forked from awslabs/aws-c-common
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbyte_buf.c
215 lines (169 loc) · 5.79 KB
/
byte_buf.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
/*
* Copyright 2010-2018 Amazon.com, Inc. or its affiliates. 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.
* A copy of the License is located at
*
* http://aws.amazon.com/apache2.0
*
* or in the "license" file accompanying this file. This file is distributed
* on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
* express or implied. See the License for the specific language governing
* permissions and limitations under the License.
*/
#include <aws/common/byte_buf.h>
#include <assert.h>
#include <stdarg.h>
#ifdef _MSC_VER
/* disables warning non const declared initializers for Microsoft compilers */
# pragma warning(disable : 4204)
# pragma warning(disable : 4706)
#endif
int aws_byte_buf_init(struct aws_allocator *allocator, struct aws_byte_buf *buf, size_t capacity) {
buf->buffer = (uint8_t *)aws_mem_acquire(allocator, capacity);
if (!buf->buffer) {
return AWS_OP_ERR;
}
buf->len = 0;
buf->capacity = capacity;
buf->allocator = allocator;
return AWS_OP_SUCCESS;
}
void aws_byte_buf_clean_up(struct aws_byte_buf *buf) {
if (buf->allocator && buf->buffer) {
aws_mem_release(buf->allocator, (void *)buf->buffer);
}
buf->allocator = NULL;
buf->buffer = NULL;
buf->len = 0;
buf->capacity = 0;
}
void aws_byte_buf_secure_zero(struct aws_byte_buf *buf) {
if (buf->buffer) {
aws_secure_zero(buf->buffer, buf->capacity);
}
buf->len = 0;
}
void aws_byte_buf_clean_up_secure(struct aws_byte_buf *buf) {
aws_byte_buf_secure_zero(buf);
aws_byte_buf_clean_up(buf);
}
bool aws_byte_buf_eq(const struct aws_byte_buf *a, const struct aws_byte_buf *b) {
if (!a || !b) {
return (a == b);
}
if (a->len != b->len) {
return false;
}
if (!a->buffer || !b->buffer) {
return (a->buffer == b->buffer);
}
return !memcmp(a->buffer, b->buffer, a->len);
}
int aws_byte_buf_init_copy(struct aws_allocator *allocator, struct aws_byte_buf *dest, const struct aws_byte_buf *src) {
assert(allocator);
assert(dest);
assert(src);
dest->len = 0;
dest->capacity = 0;
dest->allocator = NULL;
if (src->buffer == NULL) {
dest->buffer = NULL;
return AWS_OP_SUCCESS;
}
dest->buffer = (uint8_t *)aws_mem_acquire(allocator, sizeof(uint8_t) * src->len);
if (dest->buffer == NULL) {
return AWS_OP_ERR;
}
dest->len = src->len;
dest->capacity = src->len;
dest->allocator = allocator;
memcpy(dest->buffer, src->buffer, src->len);
return AWS_OP_SUCCESS;
}
int aws_byte_buf_split_on_char_n(
struct aws_byte_buf *input_str,
char split_on,
struct aws_array_list *output,
size_t n) {
assert(input_str);
assert(output);
assert(output->item_size >= sizeof(struct aws_byte_cursor));
size_t max_splits = n > 0 ? n : SIZE_MAX;
size_t last_offset = 0, split_count = 0;
uint8_t *new_location = NULL;
struct aws_byte_cursor current_pos = aws_byte_cursor_from_buf(input_str);
while (split_count < max_splits && (new_location = memchr(current_pos.ptr, split_on, current_pos.len))) {
size_t distance_from_origin = new_location - current_pos.ptr;
struct aws_byte_cursor buffer = aws_byte_cursor_advance(¤t_pos, distance_from_origin);
/* skip ahead by one to jump over the split_on character.*/
aws_byte_cursor_advance(¤t_pos, 1);
if (AWS_UNLIKELY(aws_array_list_push_back(output, (const void *)&buffer))) {
return AWS_OP_ERR;
}
last_offset = distance_from_origin + 1;
split_count += 1;
}
if (last_offset < input_str->len) {
return aws_array_list_push_back(output, (const void *)¤t_pos);
}
/* if we get here, we hit the end and we need to add an empty split */
struct aws_byte_cursor cursor = aws_byte_cursor_from_array(NULL, 0);
if (AWS_UNLIKELY(aws_array_list_push_back(output, (const void *)&cursor))) {
return AWS_OP_ERR;
}
return AWS_OP_SUCCESS;
}
int aws_byte_buf_split_on_char(struct aws_byte_buf *input_str, char split_on, struct aws_array_list *output) {
return aws_byte_buf_split_on_char_n(input_str, split_on, output, 0);
}
int aws_byte_buf_cat(struct aws_byte_buf *dest, size_t number_of_args, ...) {
assert(dest);
va_list ap;
va_start(ap, number_of_args);
for (size_t i = 0; i < number_of_args; ++i) {
struct aws_byte_buf *buffer = va_arg(ap, struct aws_byte_buf *);
struct aws_byte_cursor cursor = aws_byte_cursor_from_buf(buffer);
if (aws_byte_buf_append(dest, &cursor)) {
va_end(ap);
return AWS_OP_ERR;
}
}
va_end(ap);
return AWS_OP_SUCCESS;
}
bool aws_byte_cursor_eq(const struct aws_byte_cursor *a, const struct aws_byte_cursor *b) {
if (!a || !b) {
return (a == b);
}
if (a->len != b->len) {
return false;
}
if (!a->ptr || !b->ptr) {
return (a->ptr == b->ptr);
}
return !memcmp(a->ptr, b->ptr, a->len);
}
bool aws_byte_cursor_eq_byte_buf(const struct aws_byte_cursor *a, const struct aws_byte_buf *b) {
if (!a || !b) {
return ((void *)a == (void *)b);
}
if (a->len != b->len) {
return false;
}
if (!a->ptr || !b->buffer) {
return (a->ptr == b->buffer);
}
return !memcmp(a->ptr, b->buffer, a->len);
}
int aws_byte_buf_append(struct aws_byte_buf *to, struct aws_byte_cursor *from) {
assert(from->ptr);
assert(to->buffer);
if (to->capacity - to->len < from->len) {
return aws_raise_error(AWS_ERROR_DEST_COPY_TOO_SMALL);
}
memcpy(to->buffer + to->len, from->ptr, from->len);
to->len += from->len;
return AWS_OP_SUCCESS;
}