/* * Copyright 2014-2022 The GmSSL Project. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the License); you may * not use this file except in compliance with the License. * * http://www.apache.org/licenses/LICENSE-2.0 */ #include #include #include #include #include #include static int remove_newline(char *line) { size_t len; len = strlen(line); if (len >= 2) { if (line[len - 2] == '\r' && line[len - 1] == '\n') { line[len - 2] = line[len - 1] = 0; return 1; } } if (len) { if (line[len - 1] == '\n') { line[len - 1] = 0; return 1; } } return 0; // No newline found, might not be an error } int pem_write(FILE *fp, const char *name, const uint8_t *data, size_t datalen) { BASE64_CTX ctx; uint8_t out[168]; int inlen, outlen; if (!datalen) { error_print(); return -1; } if (datalen > INT_MAX) { error_print(); return -1; } fprintf(fp, "-----BEGIN %s-----\n", name); base64_encode_init(&ctx); while (datalen) { inlen = datalen < 48 ? (int)datalen : 48; base64_encode_update(&ctx, data, inlen, out, &outlen); fwrite(out, 1, outlen, fp); data += inlen; datalen -= inlen; } base64_encode_finish(&ctx, out, &outlen); fwrite(out, 1, outlen, fp); fprintf(fp, "-----END %s-----\n", name); return 1; } int pem_read(FILE *fp, const char *name, uint8_t *data, size_t *datalen, size_t maxlen) { char line[80]; char begin_line[80]; char end_line[80]; int len; BASE64_CTX ctx; snprintf(begin_line, sizeof(begin_line), "-----BEGIN %s-----", name); snprintf(end_line, sizeof(end_line), "-----END %s-----", name); if (feof(fp)) { error_print(); return 0; } if (!fgets(line, sizeof(line), fp)) { if (feof(fp)) { error_print(); return 0; } else { error_print(); return -1; } } remove_newline(line); if (strcmp(line, begin_line) != 0) { fprintf(stderr, "%s %d: %s\n", __FILE__, __LINE__, line); fprintf(stderr, "%s %d: %s\n", __FILE__, __LINE__, begin_line); error_print(); return -1; } *datalen = 0; base64_decode_init(&ctx); for (;;) { if (!fgets(line, sizeof(line), fp)) { error_print(); return -1; } remove_newline(line); if (strcmp(line, end_line) == 0) { break; } base64_decode_update(&ctx, (uint8_t *)line, (int)strlen(line), data, &len); data += len; *datalen += len; } base64_decode_finish(&ctx, data, &len); *datalen += len; return 1; }