/src/openssl32/crypto/err/err_prn.c
Line | Count | Source (jump to first uncovered line) |
1 | | /* |
2 | | * Copyright 1995-2021 The OpenSSL Project Authors. All Rights Reserved. |
3 | | * |
4 | | * Licensed under the Apache License 2.0 (the "License"). You may not use |
5 | | * this file except in compliance with the License. You can obtain a copy |
6 | | * in the file LICENSE in the source distribution or at |
7 | | * https://www.openssl.org/source/license.html |
8 | | */ |
9 | | |
10 | | #define OSSL_FORCE_ERR_STATE |
11 | | |
12 | | #include <stdio.h> |
13 | | #include "internal/cryptlib.h" |
14 | | #include <openssl/crypto.h> |
15 | | #include <openssl/buffer.h> |
16 | | #include <openssl/err.h> |
17 | | #include "err_local.h" |
18 | | |
19 | 22.4k | #define ERR_PRINT_BUF_SIZE 4096 |
20 | | void ERR_print_errors_cb(int (*cb) (const char *str, size_t len, void *u), |
21 | | void *u) |
22 | 78.1k | { |
23 | 78.1k | CRYPTO_THREAD_ID tid = CRYPTO_THREAD_get_current_id(); |
24 | 78.1k | unsigned long l; |
25 | 78.1k | const char *file, *data, *func; |
26 | 78.1k | int line, flags; |
27 | | |
28 | 101k | while ((l = ERR_get_error_all(&file, &line, &func, &data, &flags)) != 0) { |
29 | 23.2k | char buf[ERR_PRINT_BUF_SIZE] = ""; |
30 | 23.2k | char *hex = NULL; |
31 | 23.2k | int offset; |
32 | | |
33 | 23.2k | if ((flags & ERR_TXT_STRING) == 0) |
34 | 23.0k | data = ""; |
35 | | |
36 | 23.2k | hex = ossl_buf2hexstr_sep((const unsigned char *)&tid, sizeof(tid), '\0'); |
37 | 23.2k | BIO_snprintf(buf, sizeof(buf), "%s:", hex == NULL ? "<null>" : hex); |
38 | 23.2k | offset = strlen(buf); |
39 | 23.2k | ossl_err_string_int(l, func, buf + offset, sizeof(buf) - offset); |
40 | 23.2k | offset += strlen(buf + offset); |
41 | 23.2k | BIO_snprintf(buf + offset, sizeof(buf) - offset, ":%s:%d:%s\n", |
42 | 23.2k | file, line, data); |
43 | 23.2k | OPENSSL_free(hex); |
44 | 23.2k | if (cb(buf, strlen(buf), u) <= 0) |
45 | 0 | break; /* abort outputting the error report */ |
46 | 23.2k | } |
47 | 78.1k | } |
48 | | |
49 | | /* auxiliary function for incrementally reporting texts via the error queue */ |
50 | | static void put_error(int lib, const char *func, int reason, |
51 | | const char *file, int line) |
52 | 0 | { |
53 | 0 | ERR_new(); |
54 | 0 | ERR_set_debug(file, line, func); |
55 | 0 | ERR_set_error(lib, reason, NULL /* no data here, so fmt is NULL */); |
56 | 0 | } |
57 | | |
58 | 22.4k | #define TYPICAL_MAX_OUTPUT_BEFORE_DATA 100 |
59 | 29.9k | #define MAX_DATA_LEN (ERR_PRINT_BUF_SIZE - TYPICAL_MAX_OUTPUT_BEFORE_DATA) |
60 | | void ERR_add_error_txt(const char *separator, const char *txt) |
61 | 7.49k | { |
62 | 7.49k | const char *file = NULL; |
63 | 7.49k | int line; |
64 | 7.49k | const char *func = NULL; |
65 | 7.49k | const char *data = NULL; |
66 | 7.49k | int flags; |
67 | 7.49k | unsigned long err = ERR_peek_last_error(); |
68 | | |
69 | 7.49k | if (separator == NULL) |
70 | 4.47k | separator = ""; |
71 | 7.49k | if (err == 0) |
72 | 0 | put_error(ERR_LIB_NONE, NULL, 0, "", 0); |
73 | | |
74 | 7.49k | do { |
75 | 7.49k | size_t available_len, data_len; |
76 | 7.49k | const char *curr = txt, *next = txt; |
77 | 7.49k | const char *leading_separator = separator; |
78 | 7.49k | int trailing_separator = 0; |
79 | 7.49k | char *tmp; |
80 | | |
81 | 7.49k | ERR_peek_last_error_all(&file, &line, &func, &data, &flags); |
82 | 7.49k | if ((flags & ERR_TXT_STRING) == 0) { |
83 | 3.98k | data = ""; |
84 | 3.98k | leading_separator = ""; |
85 | 3.98k | } |
86 | 7.49k | data_len = strlen(data); |
87 | | |
88 | | /* workaround for limit of ERR_print_errors_cb() */ |
89 | 7.49k | if (data_len >= MAX_DATA_LEN |
90 | 7.49k | || strlen(separator) >= (size_t)(MAX_DATA_LEN - data_len)) |
91 | 0 | available_len = 0; |
92 | 7.49k | else |
93 | 7.49k | available_len = MAX_DATA_LEN - data_len - strlen(separator) - 1; |
94 | | /* MAX_DATA_LEN > available_len >= 0 */ |
95 | | |
96 | 7.49k | if (*separator == '\0') { |
97 | 4.47k | const size_t len_next = strlen(next); |
98 | | |
99 | 4.47k | if (len_next <= available_len) { |
100 | 4.47k | next += len_next; |
101 | 4.47k | curr = NULL; /* no need to split */ |
102 | 4.47k | } else { |
103 | 0 | next += available_len; |
104 | 0 | curr = next; /* will split at this point */ |
105 | 0 | } |
106 | 4.47k | } else { |
107 | 45.4k | while (*next != '\0' && (size_t)(next - txt) <= available_len) { |
108 | 42.4k | curr = next; |
109 | 42.4k | next = strstr(curr, separator); |
110 | 42.4k | if (next != NULL) { |
111 | 41.5k | next += strlen(separator); |
112 | 41.5k | trailing_separator = *next == '\0'; |
113 | 41.5k | } else { |
114 | 848 | next = curr + strlen(curr); |
115 | 848 | } |
116 | 42.4k | } |
117 | 3.02k | if ((size_t)(next - txt) <= available_len) |
118 | 3.02k | curr = NULL; /* the above loop implies *next == '\0' */ |
119 | 3.02k | } |
120 | 7.49k | if (curr != NULL) { |
121 | | /* split error msg at curr since error data would get too long */ |
122 | 0 | if (curr != txt) { |
123 | 0 | tmp = OPENSSL_strndup(txt, curr - txt); |
124 | 0 | if (tmp == NULL) |
125 | 0 | return; |
126 | 0 | ERR_add_error_data(2, separator, tmp); |
127 | 0 | OPENSSL_free(tmp); |
128 | 0 | } |
129 | 0 | put_error(ERR_GET_LIB(err), func, err, file, line); |
130 | 0 | txt = curr; |
131 | 7.49k | } else { |
132 | 7.49k | if (trailing_separator) { |
133 | 2.17k | tmp = OPENSSL_strndup(txt, next - strlen(separator) - txt); |
134 | 2.17k | if (tmp == NULL) |
135 | 0 | return; |
136 | | /* output txt without the trailing separator */ |
137 | 2.17k | ERR_add_error_data(2, leading_separator, tmp); |
138 | 2.17k | OPENSSL_free(tmp); |
139 | 5.32k | } else { |
140 | 5.32k | ERR_add_error_data(2, leading_separator, txt); |
141 | 5.32k | } |
142 | 7.49k | txt = next; /* finished */ |
143 | 7.49k | } |
144 | 7.49k | } while (*txt != '\0'); |
145 | 7.49k | } |
146 | | |
147 | | void ERR_add_error_mem_bio(const char *separator, BIO *bio) |
148 | 2.17k | { |
149 | 2.17k | if (bio != NULL) { |
150 | 2.17k | char *str; |
151 | 2.17k | long len = BIO_get_mem_data(bio, &str); |
152 | | |
153 | 2.17k | if (len > 0) { |
154 | 2.17k | if (str[len - 1] != '\0') { |
155 | 2.17k | if (BIO_write(bio, "", 1) <= 0) |
156 | 0 | return; |
157 | | |
158 | 2.17k | len = BIO_get_mem_data(bio, &str); |
159 | 2.17k | } |
160 | 2.17k | if (len > 1) |
161 | 2.17k | ERR_add_error_txt(separator, str); |
162 | 2.17k | } |
163 | 2.17k | } |
164 | 2.17k | } |
165 | | |
166 | | static int print_bio(const char *str, size_t len, void *bp) |
167 | 23.2k | { |
168 | 23.2k | return BIO_write((BIO *)bp, str, len); |
169 | 23.2k | } |
170 | | |
171 | | void ERR_print_errors(BIO *bp) |
172 | 22.7k | { |
173 | 22.7k | ERR_print_errors_cb(print_bio, bp); |
174 | 22.7k | } |
175 | | |
176 | | #ifndef OPENSSL_NO_STDIO |
177 | | void ERR_print_errors_fp(FILE *fp) |
178 | 55.4k | { |
179 | 55.4k | BIO *bio = BIO_new_fp(fp, BIO_NOCLOSE); |
180 | 55.4k | if (bio == NULL) |
181 | 0 | return; |
182 | | |
183 | 55.4k | ERR_print_errors_cb(print_bio, bio); |
184 | 55.4k | BIO_free(bio); |
185 | 55.4k | } |
186 | | #endif |