Mercurial > libedl
comparison src/edl.c @ 3:bd99b6549eb4
*: expand in readme and rename src/main to src/edl
author | Paper <mrpapersonic@gmail.com> |
---|---|
date | Sun, 24 Dec 2023 20:22:54 -0500 (13 months ago) |
parents | |
children | c2408abb258a |
comparison
equal
deleted
inserted
replaced
2:d00bc412900e | 3:bd99b6549eb4 |
---|---|
1 #include <string.h> | |
2 #include <stdbool.h> | |
3 #include <stdio.h> | |
4 #include <stdlib.h> | |
5 #include <stdint.h> | |
6 #include "edl.h" | |
7 | |
8 #define strnchr(haystack, needle, length) (char*)memchr(haystack, needle, strnlen(haystack, length)) | |
9 | |
10 static uint32_t EDL_internal_crcn32b(const unsigned char* restrict message, size_t length) { | |
11 uint32_t crc = 0xFFFFFFFF; | |
12 | |
13 size_t i; | |
14 for (i = 0; i < length && message[i]; i++) { | |
15 crc = crc ^ message[i]; | |
16 size_t j; | |
17 for (j = 0; j < 8; j++) | |
18 crc = (crc >> 1) ^ (0xEDB88320 & -(crc & 1)); | |
19 } | |
20 | |
21 return ~crc; | |
22 } | |
23 | |
24 /* This parses the first line of the EDL and | |
25 * creates an array with CRC32 hashes in | |
26 * the order of the strings. | |
27 */ | |
28 static uint32_t* EDL_internal_parse_first_line(const char* data, size_t* restrict length) { | |
29 size_t hashes_size = 32, current_hash = 0; | |
30 uint32_t* hashes = malloc(hashes_size * sizeof(uint32_t)); | |
31 | |
32 size_t len_until_newline = strchr(data, '\n') - data, i, b; | |
33 for (i = 0, b = 0; i < len_until_newline; i++, b++) { | |
34 if (data[i] == ';') { | |
35 if (current_hash >= hashes_size) | |
36 hashes = realloc(hashes, (hashes_size *= 2) * sizeof(uint32_t)); | |
37 hashes[current_hash++] = EDL_internal_crcn32b((const unsigned char*)&data[i - b], b); | |
38 b = -1; // ew | |
39 } | |
40 } | |
41 | |
42 *length = current_hash; | |
43 | |
44 return hashes; | |
45 } | |
46 | |
47 /* -- Functions to extract different datatypes -- */ | |
48 | |
49 static size_t EDL_internal_append_offset(const char* input, size_t offset, size_t length) { | |
50 size_t s = 0; | |
51 | |
52 s += strchr(&input[offset + s], ';') - &input[offset]; | |
53 if (s + offset > length) | |
54 return s; | |
55 | |
56 s += strspn(&input[offset + s], ";\t "); | |
57 if (s + offset > length) | |
58 return s; | |
59 | |
60 return s; | |
61 } | |
62 | |
63 static size_t EDL_internal_get_int(const char* input, size_t offset, size_t length, int* int_return) { | |
64 if (offset > length) | |
65 return 0; | |
66 | |
67 { | |
68 char* ptr_return; | |
69 *int_return = strtol(&input[offset], &ptr_return, 10); | |
70 | |
71 if (!ptr_return) | |
72 return 0; | |
73 } | |
74 | |
75 return EDL_internal_append_offset(input, offset, length); | |
76 } | |
77 | |
78 static size_t EDL_internal_get_double(const char* input, size_t offset, size_t length, double* double_return) { | |
79 if (offset > length) | |
80 return 0; | |
81 | |
82 { | |
83 char* ptr_return; | |
84 *double_return = strtod(&input[offset], &ptr_return); | |
85 | |
86 if (!ptr_return) | |
87 return 0; | |
88 } | |
89 | |
90 return EDL_internal_append_offset(input, offset, length); | |
91 } | |
92 | |
93 static size_t EDL_internal_get_bool(const char* input, size_t offset, size_t length, bool* bool_return) { | |
94 if (offset > length) | |
95 return 0; | |
96 | |
97 if (strncmp((input + offset), "TRUE", 4)) | |
98 *bool_return = true; | |
99 else if (strncmp((input + offset), "FALSE", 5)) | |
100 *bool_return = false; | |
101 | |
102 return EDL_internal_append_offset(input, offset, length); | |
103 } | |
104 | |
105 static size_t EDL_internal_get_media_type(const char* input, size_t offset, size_t length, MediaType* media_return) { | |
106 if (offset > length) | |
107 return 0; | |
108 | |
109 if (strncmp(input + offset, "VIDEO", 5)) | |
110 *media_return = MEDIATYPE_VIDEO; | |
111 else if (strncmp(input + offset, "AUDIO", 5)) | |
112 *media_return = MEDIATYPE_AUDIO; | |
113 | |
114 return EDL_internal_append_offset(input, offset, length); | |
115 } | |
116 | |
117 static size_t EDL_internal_get_string(const char* input, size_t offset, size_t length, char** string_return) { | |
118 /* Windows filenames will *NEVER* include double quotes. | |
119 This might break with EDL files created with Reaper on Linux. */ | |
120 if (offset > length) | |
121 return 0; | |
122 | |
123 size_t start = strchr(&input[offset], '\"') - &input[offset] + 1; | |
124 if (start + offset > length) | |
125 return 0; | |
126 | |
127 size_t s_length = strchr(&input[offset + start], '\"') - &input[offset] - start; | |
128 if (start + s_length + offset > length) | |
129 return 0; | |
130 | |
131 *string_return = malloc((s_length + 1) * sizeof(char)); | |
132 if (!*string_return) | |
133 return 0; | |
134 | |
135 memcpy(*string_return, &input[offset + start], s_length); | |
136 (*string_return)[s_length] = '\0'; | |
137 | |
138 return EDL_internal_append_offset(input, offset, length); | |
139 } | |
140 | |
141 /* memory management routines */ | |
142 | |
143 static int EDL_internal_reallocate(EDL_file* input, size_t new_capacity) { | |
144 input->arr = realloc(input->arr, new_capacity * sizeof(EDL_line)); | |
145 if (!input->arr) | |
146 return 0; | |
147 | |
148 if (new_capacity > input->capacity) | |
149 memset(input->arr + input->capacity, 0, new_capacity - input->capacity); | |
150 | |
151 input->capacity = new_capacity; | |
152 | |
153 return 1; | |
154 } | |
155 | |
156 static int EDL_internal_allocate_memory(EDL_file* input) { | |
157 return EDL_internal_reallocate(input, input->capacity ? input->capacity * 2 : 16); | |
158 } | |
159 | |
160 static int EDL_internal_shrink_memory_to_fit(EDL_file* input) { | |
161 return EDL_internal_reallocate(input, input->size); | |
162 } | |
163 | |
164 /* the important function */ | |
165 EDL_file EDL_parse(const char* data, size_t length) { | |
166 EDL_file edl = {0}; | |
167 if (!EDL_internal_allocate_memory(&edl)) | |
168 return edl; | |
169 | |
170 size_t order_size = 0; | |
171 uint32_t* order = EDL_internal_parse_first_line(data, &order_size); | |
172 | |
173 size_t offset = strnchr(data, '\n', length) - data + 1; | |
174 while ((offset = strnchr(data + offset, '\n', length - offset) - data + 1) < length) { | |
175 size_t local_offset = offset; // this is so our original offset stays intact | |
176 | |
177 bool ok = true; | |
178 | |
179 for (int i = 0; i < order_size; i++) { | |
180 #define ADD_TO_OFFSET(x, a) \ | |
181 { \ | |
182 size_t o = a(data, local_offset, length, &edl.arr[edl.size].x); \ | |
183 if (!o) \ | |
184 ok = false; \ | |
185 local_offset += o; \ | |
186 } | |
187 switch (order[i]) { | |
188 case 0x1541c503: /* ID */ | |
189 ADD_TO_OFFSET(id, EDL_internal_get_int); | |
190 break; | |
191 case 0x4b211812: /* Track */ | |
192 ADD_TO_OFFSET(track, EDL_internal_get_int); | |
193 break; | |
194 case 0xbb46516f: /* StartTime */ | |
195 ADD_TO_OFFSET(start_time, EDL_internal_get_double); | |
196 break; | |
197 case 0xaeac5df7: /* Length */ | |
198 ADD_TO_OFFSET(length, EDL_internal_get_double); | |
199 break; | |
200 case 0x03834606: /* PlayRate */ | |
201 ADD_TO_OFFSET(play_rate, EDL_internal_get_double); | |
202 break; | |
203 case 0x0c2083d3: /* Locked */ | |
204 ADD_TO_OFFSET(locked, EDL_internal_get_bool); | |
205 break; | |
206 case 0xe60c8b1d: /* Normalized */ | |
207 ADD_TO_OFFSET(normalized, EDL_internal_get_bool); | |
208 break; | |
209 case 0xbe5802c9: /* StretchMethod */ | |
210 ADD_TO_OFFSET(stretch_method, EDL_internal_get_int); | |
211 break; | |
212 case 0x4ec8be4c: /* Looped */ | |
213 ADD_TO_OFFSET(looped, EDL_internal_get_bool); | |
214 break; | |
215 case 0xe6cb84d1: /* OnRuler */ | |
216 ADD_TO_OFFSET(on_ruler, EDL_internal_get_bool); | |
217 break; | |
218 case 0x035f84c2: /* MediaType */ | |
219 ADD_TO_OFFSET(media_type, EDL_internal_get_media_type); | |
220 break; | |
221 case 0x379d32c5: /* FileName */ | |
222 ADD_TO_OFFSET(filename, EDL_internal_get_string); | |
223 break; | |
224 case 0x9f334738: /* Stream */ | |
225 ADD_TO_OFFSET(stream, EDL_internal_get_int); | |
226 break; | |
227 case 0x07b4e0e7: /* StreamStart */ | |
228 ADD_TO_OFFSET(stream_start, EDL_internal_get_double); | |
229 break; | |
230 case 0x8f16c7b8: /* StreamLength */ | |
231 ADD_TO_OFFSET(stream_length, EDL_internal_get_double); | |
232 break; | |
233 case 0xd2edd7e4: /* FadeTimeIn */ | |
234 ADD_TO_OFFSET(fade_time_in, EDL_internal_get_double); | |
235 break; | |
236 case 0x792e8c40: /* FadeTimeOut */ | |
237 ADD_TO_OFFSET(fade_time_out, EDL_internal_get_double); | |
238 break; | |
239 case 0x98374657: /* SustainGain */ | |
240 ADD_TO_OFFSET(sustain_gain, EDL_internal_get_double); | |
241 break; | |
242 case 0x3e998b1f: /* CurveIn */ | |
243 ADD_TO_OFFSET(curve_in, EDL_internal_get_int); | |
244 break; | |
245 case 0x82fb09c4: /* GainIn */ | |
246 ADD_TO_OFFSET(gain_in, EDL_internal_get_double); | |
247 break; | |
248 case 0x53add388: /* CurveOut */ | |
249 ADD_TO_OFFSET(curve_out, EDL_internal_get_int); | |
250 break; | |
251 case 0x4210ba56: /* GainOut */ | |
252 ADD_TO_OFFSET(gain_out, EDL_internal_get_double); | |
253 break; | |
254 case 0x89c4df6c: /* Layer */ | |
255 ADD_TO_OFFSET(layer, EDL_internal_get_int); | |
256 break; | |
257 case 0xadf2f1a3: /* Color */ | |
258 ADD_TO_OFFSET(color, EDL_internal_get_int); | |
259 break; | |
260 case 0xa56b43e1: /* CurveInR */ | |
261 ADD_TO_OFFSET(curve_in_r, EDL_internal_get_int); | |
262 break; | |
263 case 0xcb6d715e: /* CurveOutR */ | |
264 ADD_TO_OFFSET(curve_out_r, EDL_internal_get_int); | |
265 break; | |
266 case 0x9da1b9ed: /* PlayPitch */ | |
267 ADD_TO_OFFSET(play_pitch, EDL_internal_get_double); | |
268 break; | |
269 case 0x2bda6ed4: /* LockPitch */ | |
270 ADD_TO_OFFSET(lock_pitch, EDL_internal_get_bool); | |
271 break; | |
272 case 0x3071a4c6: /* FirstChannel */ | |
273 ADD_TO_OFFSET(first_channel, EDL_internal_get_int); | |
274 break; | |
275 case 0x7de1bd40: /* Channels */ | |
276 ADD_TO_OFFSET(channels, EDL_internal_get_int); | |
277 break; | |
278 default: | |
279 /* ... what */ | |
280 break; | |
281 } | |
282 } | |
283 | |
284 if (!ok) | |
285 break; | |
286 | |
287 if (++edl.size >= edl.capacity) | |
288 EDL_internal_allocate_memory(&edl); | |
289 } | |
290 | |
291 EDL_internal_shrink_memory_to_fit(&edl); | |
292 | |
293 free(order); | |
294 | |
295 return edl; | |
296 } | |
297 | |
298 void EDL_free(EDL_file file) { | |
299 size_t i; | |
300 for (i = 0; i < file.size; i++) { | |
301 if (file.arr[i].filename) | |
302 free(file.arr[i].filename); | |
303 } | |
304 free(file.arr); | |
305 } |