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 |
| 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 } |
