103 const uint8_t *data, uint8_t sector_status, uint32_t length)
105 TRACE(
"Entering aaruf_write_sector(%p, %" PRIu64
", %d, %p, %u, %u)", context, sector_address, negative, data,
106 sector_status, length);
111 FATAL(
"Invalid context");
113 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_AARUFORMAT");
122 FATAL(
"Invalid context");
124 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_AARUFORMAT");
131 FATAL(
"Trying to write a read-only image");
133 TRACE(
"Exiting aaruf_write_sector() = AARUF_READ_ONLY");
139 FATAL(
"Sector address out of bounds");
141 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_SECTOR_OUT_OF_BOUNDS");
147 FATAL(
"Sector address out of bounds");
149 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_SECTOR_OUT_OF_BOUNDS");
153 if(length > USHRT_MAX)
155 FATAL(
"Sector length too large");
157 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_INVALID_SECTOR_LENGTH");
168 TRACE(
"NGCW sector detected, disabling checksums");
177 else if(sector_address <= ctx->last_written_block)
221 !negative && sector_address <= ctx->image_info.Sectors && !ctx->
writing_long)
226 const uint8_t *write_data = data;
227 uint8_t *decrypted_buffer = NULL;
241 decrypted_buffer = (uint8_t *)malloc(length);
243 if(decrypted_buffer == NULL)
245 FATAL(
"Could not allocate memory for PS3 decryption buffer");
249 memcpy(decrypted_buffer, data, length);
251 write_data = decrypted_buffer;
258 uint64_t ddt_entry_gen = 0;
259 bool ddt_ok_gen =
set_ddt_entry_v2(ctx, sector_address, negative, 0, 0, sector_status, &ddt_entry_gen);
261 free(decrypted_buffer);
265 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_CANNOT_SET_DDT_ENTRY");
269 TRACE(
"Exiting aaruf_write_sector() = AARUF_STATUS_OK (generable)");
279 TRACE(
"Closing current block before writing new data");
284 FATAL(
"Error closing current block: %d", error);
286 TRACE(
"Exiting aaruf_write_sector() = %d", error);
291 uint64_t ddt_entry = 0;
297 TRACE(
"Hashing sector data for deduplication");
298 uint64_t hash = XXH3_64bits(write_data, length);
302 TRACE(
"Block does %s exist in deduplication map", existing ?
"already" :
"not yet");
305 sector_status, &ddt_entry);
308 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_CANNOT_SET_DDT_ENTRY");
314 TRACE(
"Sector exists, so not writing to image");
315 free(decrypted_buffer);
316 TRACE(
"Exiting aaruf_write_sector() = AARUF_STATUS_OK");
320 TRACE(
"Inserting sector hash into deduplication map, proceeding to write into image as normal");
325 sector_status, &ddt_entry);
329 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_CANNOT_SET_DDT_ENTRY");
336 TRACE(
"Creating new writing block");
346 if(sector_address >= ctx->
track_entries[i].
start && sector_address <= ctx->track_entries[i].end)
389 uint32_t max_buffer_size =
391 TRACE(
"Setting max buffer size to %u bytes", max_buffer_size);
393 TRACE(
"Allocating memory for writing buffer");
397 FATAL(
"Could not allocate memory");
399 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
411 free(decrypted_buffer);
413 TRACE(
"Exiting aaruf_write_sector() = AARUF_STATUS_OK");
624 const uint8_t *data, uint8_t sector_status, uint32_t length)
626 TRACE(
"Entering aaruf_write_sector_long(%p, %" PRIu64
", %d, %p, %u, %u)", context, sector_address, negative, data,
627 sector_status, length);
632 FATAL(
"Invalid context");
634 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_AARUFORMAT");
643 FATAL(
"Invalid context");
645 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_AARUFORMAT");
652 FATAL(
"Trying to write a read-only image");
654 TRACE(
"Exiting aaruf_write_sector() = AARUF_READ_ONLY");
660 FATAL(
"Sector address out of bounds");
662 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_SECTOR_OUT_OF_BOUNDS");
668 FATAL(
"Sector address out of bounds");
670 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_SECTOR_OUT_OF_BOUNDS");
681 if(sector_address >= ctx->
track_entries[i].
start && sector_address <= ctx->track_entries[i].end)
689 uint64_t corrected_sector_address = sector_address;
697 uint64_t total_sectors =
715 memcpy(ctx->
sector_id + corrected_sector_address * 4, data, 4);
716 memcpy(ctx->
sector_ied + corrected_sector_address * 2, data + 4, 2);
717 memcpy(ctx->
sector_cpr_mai + corrected_sector_address * 6, data + 6, 6);
718 memcpy(ctx->
sector_edc + corrected_sector_address * 4, data + 2060, 4);
720 return aaruf_write_sector(context, sector_address, negative, data + 12, sector_status, 2048);
731 memcpy(ctx->
sector_id + corrected_sector_address * 4, data, 4);
732 memcpy(ctx->
sector_ied + corrected_sector_address * 2, data + 4, 2);
733 memcpy(ctx->
sector_cpr_mai + corrected_sector_address * 6, data + 2054, 6);
734 memcpy(ctx->
sector_edc + corrected_sector_address * 4, data + 2060, 4);
736 return aaruf_write_sector(context, sector_address, negative, data + 6, sector_status, 2048);
748 memcpy(ctx->
sector_edc + corrected_sector_address * 4, data + 2048, 4);
750 return aaruf_write_sector(context, sector_address, negative, data, sector_status, 2048);
755 FATAL(
"Incorrect sector size");
756 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_INCORRECT_DATA_SIZE");
764 if(sector_address <= ctx->last_written_block)
790 if(ctx->
calculating_md5 && !negative && sector_address <= ctx->image_info.Sectors)
793 if(ctx->
calculating_sha1 && !negative && sector_address <= ctx->image_info.Sectors)
812 return aaruf_write_sector(context, sector_address, negative, data, sector_status, length);
824 FATAL(
"Could not allocate memory for CD sector prefix DDT");
826 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
840 FATAL(
"Could not allocate memory for CD sector prefix DDT");
842 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
855 FATAL(
"Could not allocate memory for CD sector prefix buffer");
857 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
871 FATAL(
"Could not allocate memory for CD sector suffix buffer");
873 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
880 for(
int i = 0; i < length; i++)
897 prefix_correct =
true;
899 if(data[0x00] != 0x00 || data[0x01] != 0xFF || data[0x02] != 0xFF || data[0x03] != 0xFF ||
900 data[0x04] != 0xFF || data[0x05] != 0xFF || data[0x06] != 0xFF || data[0x07] != 0xFF ||
901 data[0x08] != 0xFF || data[0x09] != 0xFF || data[0x0A] != 0xFF || data[0x0B] != 0x00 ||
903 prefix_correct =
false;
907 const int minute = (data[0x0C] >> 4) * 10 + (data[0x0C] & 0x0F);
908 const int second = (data[0x0D] >> 4) * 10 + (data[0x0D] & 0x0F);
909 const int frame = (data[0x0E] >> 4) * 10 + (data[0x0E] & 0x0F);
910 const int stored_lba = minute * 60 * 75 + second * 75 + frame - 150;
911 prefix_correct = stored_lba == sector_address;
933 FATAL(
"Could not allocate memory for CD sector prefix buffer");
935 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
963 FATAL(
"Could not allocate memory for CD sector suffix buffer");
965 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
986 FATAL(
"Could not allocate memory for CD sector prefix DDT");
988 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
1002 FATAL(
"Could not allocate memory for CD sector prefix DDT");
1004 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
1017 FATAL(
"Could not allocate memory for CD sector prefix buffer");
1019 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
1033 FATAL(
"Could not allocate memory for CD sector suffix buffer");
1035 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
1042 for(
int i = 0; i < length; i++)
1057 const bool form2 = (data[18] & 0x20) == 0x20 || (data[22] & 0x20) == 0x20;
1059 prefix_correct =
true;
1061 if(data[0x00] != 0x00 || data[0x01] != 0xFF || data[0x02] != 0xFF || data[0x03] != 0xFF ||
1062 data[0x04] != 0xFF || data[0x05] != 0xFF || data[0x06] != 0xFF || data[0x07] != 0xFF ||
1063 data[0x08] != 0xFF || data[0x09] != 0xFF || data[0x0A] != 0xFF || data[0x0B] != 0x00 ||
1065 prefix_correct =
false;
1069 const int minute = (data[0x0C] >> 4) * 10 + (data[0x0C] & 0x0F);
1070 const int second = (data[0x0D] >> 4) * 10 + (data[0x0D] & 0x0F);
1071 const int frame = (data[0x0E] >> 4) * 10 + (data[0x0E] & 0x0F);
1072 const int stored_lba = minute * 60 * 75 + second * 75 + frame - 150;
1073 prefix_correct = stored_lba == sector_address;
1096 FATAL(
"Could not allocate memory for CD sector prefix buffer");
1098 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
1113 FATAL(
"Could not allocate memory for CD mode 2 subheader buffer");
1115 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
1124 memcpy(&edc, data + 0x92C,
sizeof(edc));
1125 const bool correct_edc = computed_edc == edc;
1151 FATAL(
"Could not allocate memory for CD sector suffix buffer");
1153 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
1160 memcpy(ctx->
mode2_subheaders + corrected_sector_address * 8, data + 0x10, 8);
1174 memcpy(&edc, data + 0x818,
sizeof(edc));
1175 const bool correct_edc = computed_edc == edc;
1177 if(correct_ecc && correct_edc)
1196 FATAL(
"Could not allocate memory for CD sector suffix buffer");
1198 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
1205 memcpy(ctx->
mode2_subheaders + corrected_sector_address * 8, data + 0x10, 8);
1210 context, sector_address, negative, data + 24,
1229 switch(length - 512)
1251 newTag = malloc(12);
1252 memcpy(newTag, data + 512, 12);
1264 newTag = malloc(20);
1265 memcpy(newTag, data + 512, 20);
1295 newTag = malloc(24);
1296 memcpy(newTag, data + 512, 24);
1312 FATAL(
"Incorrect sector size");
1313 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_INCORRECT_DATA_SIZE");
1318 return aaruf_write_sector(context, sector_address, negative, data, sector_status, 512);
1327 FATAL(
"Could not allocate memory for sector subchannel DDT");
1331 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
1336 memcpy(ctx->
sector_subchannel + sector_address * newTagSize, newTag, newTagSize);
1340 return aaruf_write_sector(context, sector_address, negative, data, sector_status, 512);
1346 TRACE(
"Exiting aaruf_write_sector() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
1546 TRACE(
"Initializing CRC64 context");
1548 TRACE(
"Updating CRC64");
1553 uint8_t *cmp_buffer = NULL;
1561 if(cmp_buffer == NULL)
1563 FATAL(
"Could not allocate buffer for compressed data");
1572 const long remaining = current_samples % flac_block_size;
1580 flac_block_size,
true,
false,
"hamming", 12, 15,
true,
false, 0, 8,
"Aaru", 4);
1591 if(cmp_buffer == NULL)
1593 FATAL(
"Could not allocate buffer for compressed data");
1613 if(cmp_buffer == NULL)
1615 FATAL(
"Could not allocate buffer for compressed data");
1619 size_t zstd_dst_size =
1623 if(zstd_dst_size == 0)
1645 FATAL(
"Invalid compression type");
1661 TRACE(
"Adding block to index");
1669 TRACE(
"Block added to index at offset %" PRIu64, index_entry.
offset);
1710 const uint8_t *ec_payload;
1711 uint32_t ec_payload_size;
1712 const uint8_t *ec_lzma_props = NULL;
1721 ec_payload = cmp_buffer;
1723 ec_lzma_props = lzma_properties;
1727 ec_payload = cmp_buffer;
1732 ec_payload, ec_payload_size, index_entry.
offset);
2004 const uint32_t length)
2006 TRACE(
"Entering aaruf_write_media_tag(%p, %p, %d, %d)", context, data, type, length);
2011 FATAL(
"Invalid context");
2013 TRACE(
"Exiting aaruf_write_media_tag() = AARUF_ERROR_NOT_AARUFORMAT");
2022 FATAL(
"Invalid context");
2024 TRACE(
"Exiting aaruf_write_media_tag() = AARUF_ERROR_NOT_AARUFORMAT");
2031 FATAL(
"Trying to write a read-only image");
2033 TRACE(
"Exiting aaruf_write_media_tag() = AARUF_READ_ONLY");
2037 if(data == NULL || length == 0)
2039 FATAL(
"Invalid data or length");
2043 uint8_t *new_data = malloc(length);
2045 if(new_data == NULL)
2047 FATAL(
"Could not allocate memory for media tag");
2050 memcpy(new_data, data, length);
2055 if(media_tag == NULL)
2057 TRACE(
"Cannot allocate memory for media tag entry.");
2064 media_tag->
type = type;
2065 media_tag->
data = new_data;
2066 media_tag->
length = length;
2068 HASH_REPLACE_INT(ctx->
mediaTags, type, media_tag, old_media_tag);
2070 if(old_media_tag != NULL)
2072 TRACE(
"Replaced media tag with type %d", old_media_tag->
type);
2073 free(old_media_tag->
data);
2074 free(old_media_tag);
2075 old_media_tag = NULL;
2079 TRACE(
"Exiting aaruf_write_media_tag() = AARUF_STATUS_OK");
2274 const uint8_t *data,
const size_t length,
const int32_t tag)
2276 TRACE(
"Entering aaruf_write_sector_tag(%p, %" PRIu64
", %d, %p, %zu, %d)", context, sector_address, negative, data,
2282 FATAL(
"Invalid context");
2284 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_AARUFORMAT");
2293 FATAL(
"Invalid context");
2295 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_AARUFORMAT");
2302 FATAL(
"Trying to write a read-only image");
2304 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_READ_ONLY");
2310 FATAL(
"Sector address out of bounds");
2312 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_SECTOR_OUT_OF_BOUNDS");
2318 FATAL(
"Sector address out of bounds");
2320 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_SECTOR_OUT_OF_BOUNDS");
2324 if(data == NULL || length == 0)
2326 FATAL(
"Invalid data or length");
2330 uint64_t corrected_sector_address = sector_address;
2338 const uint64_t total_sectors =
2346 FATAL(
"Invalid media type for tag");
2347 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2353 FATAL(
"Incorrect tag size");
2354 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2359 if(sector_address >= ctx->
track_entries[i].
start && sector_address <= ctx->track_entries[i].end)
2363 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2367 FATAL(
"Track not found");
2372 FATAL(
"Invalid media type for tag");
2373 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2379 FATAL(
"Incorrect tag size");
2380 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2385 if(sector_address >= ctx->
track_entries[i].
start && sector_address <= ctx->track_entries[i].end)
2389 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2393 FATAL(
"Track not found");
2398 FATAL(
"Invalid media type for tag");
2399 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2405 FATAL(
"Incorrect tag size");
2406 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2414 FATAL(
"Could not allocate memory for sector subchannel");
2416 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2422 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2427 FATAL(
"Invalid media type for tag");
2428 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2434 FATAL(
"Incorrect tag size");
2435 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2443 FATAL(
"Could not allocate memory for sector CPR/MAI");
2445 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2449 memcpy(ctx->
sector_cpr_mai + corrected_sector_address * 6, data, 1);
2451 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2456 FATAL(
"Invalid media type for tag");
2457 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2463 FATAL(
"Incorrect tag size");
2464 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2472 FATAL(
"Could not allocate memory for sector ID");
2474 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2478 memcpy(ctx->
sector_id + corrected_sector_address * 4, data, 1);
2479 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2484 FATAL(
"Invalid media type for tag");
2485 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2491 FATAL(
"Incorrect tag size");
2492 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2500 FATAL(
"Could not allocate memory for sector ID");
2502 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2506 memcpy(ctx->
sector_id + corrected_sector_address * 4 + 1, data, 3);
2508 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2513 FATAL(
"Invalid media type for tag");
2514 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2520 FATAL(
"Incorrect tag size");
2521 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2529 FATAL(
"Could not allocate memory for sector IED");
2531 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2535 memcpy(ctx->
sector_ied + corrected_sector_address * 2, data, 2);
2537 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2542 FATAL(
"Invalid media type for tag");
2543 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2549 FATAL(
"Incorrect tag size");
2550 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2558 FATAL(
"Could not allocate memory for sector EDC");
2560 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2564 memcpy(ctx->
sector_edc + corrected_sector_address * 4, data, 4);
2566 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2571 FATAL(
"Invalid media type for tag");
2572 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2578 FATAL(
"Incorrect tag size");
2579 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2587 FATAL(
"Could not allocate memory for sector decrypted title key");
2589 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2595 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2600 FATAL(
"Invalid media type for tag");
2601 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2607 FATAL(
"Incorrect tag size");
2608 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2616 FATAL(
"Could not allocate memory for Apple Sony tag");
2618 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2624 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2629 FATAL(
"Invalid media type for tag");
2630 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2636 FATAL(
"Incorrect tag size");
2637 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2645 FATAL(
"Could not allocate memory for Apple Profile tag");
2647 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2653 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2658 FATAL(
"Invalid media type for tag");
2659 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_MEDIA_TYPE");
2665 FATAL(
"Incorrect tag size");
2666 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_INCORRECT_DATA_SIZE");
2674 FATAL(
"Could not allocate memory for Priam Data Tower tag");
2676 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_ERROR_NOT_ENOUGH_MEMORY");
2682 TRACE(
"Exiting aaruf_write_sector_tag() = AARUF_STATUS_OK");
2685 TRACE(
"Do not know how to write sector tag %d", tag);
#define MAX_FLAKE_BLOCK
FLAC maximum block size used for encoding audio sectors.
#define LZMA_PROPERTIES_LENGTH
Size in bytes of the fixed LZMA properties header (lc/lp/pb + dictionary size).
#define AARU_MAGIC
Magic identifier for AaruFormat container (ASCII "AARUFRMT").
#define MIN_FLAKE_BLOCK
FLAC minimum block size.
#define SAMPLES_PER_SECTOR
Red Book (CD‑DA) PCM samples per 2352‑byte sector: 44,100 Hz / 75 sectors per second = 588 samples.
int32_t aaruf_lzma_encode_buffer(uint8_t *dst_buffer, size_t *dst_size, const uint8_t *src_buffer, size_t src_size, uint8_t *out_props, size_t *out_props_size, int32_t level, uint32_t dict_size, int32_t lc, int32_t lp, int32_t pb, int32_t fb, int32_t num_threads)
Encodes a buffer using LZMA compression.
uint64_t aaruf_crc64_data(const uint8_t *data, uint32_t len)
size_t aaruf_flac_encode_redbook_buffer(uint8_t *dst_buffer, size_t dst_size, const uint8_t *src_buffer, size_t src_size, uint32_t blocksize, int32_t do_mid_side_stereo, int32_t loose_mid_side_stereo, const char *apodization, uint32_t max_lpc_order, uint32_t qlp_coeff_precision, int32_t do_qlp_coeff_prec_search, int32_t do_exhaustive_model_search, uint32_t min_residual_partition_order, uint32_t max_residual_partition_order, const char *application_id, uint32_t application_id_len)
Encodes a Red Book audio buffer to FLAC format.
int aaruf_crc64_update(crc64_ctx *ctx, const uint8_t *data, uint32_t len)
Updates the CRC64 context with new data.
size_t aaruf_zstd_encode_buffer(uint8_t *dst_buffer, size_t dst_size, const uint8_t *src_buffer, size_t src_size, int level, int num_threads)
Encodes a buffer using Zstandard compression.
void aaruf_crc64_free(crc64_ctx *ctx)
Frees a CRC64 context.
int aaruf_spamsum_update(spamsum_ctx *ctx, const uint8_t *data, uint32_t len)
Updates the spamsum context with new data.
crc64_ctx * aaruf_crc64_init()
Initializes a CRC64 context.
uint32_t aaruf_edc_cd_compute(void *context, uint32_t edc, const uint8_t *src, int size, int pos)
Computes the EDC (Error Detection Code) for a CD sector.
void aaruf_md5_update(md5_ctx *ctx, const void *data, unsigned long size)
void aaruf_sha256_update(sha256_ctx *ctx, const void *data, unsigned long size)
bool aaruf_ecc_cd_is_suffix_correct_mode2(void *context, const uint8_t *sector)
Checks if the suffix (EDC/ECC) of a CD sector is correct (Mode 2).
void aaruf_sha1_update(sha1_ctx *ctx, const void *data, unsigned long size)
int aaruf_crc64_final(crc64_ctx *ctx, uint64_t *crc)
Computes the final CRC64 value from the context.
bool aaruf_ecc_cd_is_suffix_correct(void *context, const uint8_t *sector)
Checks if the suffix (EDC/ECC) of a CD sector is correct (Mode 1).
@ DataBlock
Block containing data.
@ SectorStatusNotDumped
Sector(s) not yet acquired during image dumping.
@ SectorStatusGenerable
Content can be generated using a known algorithm.
@ SectorStatusUnencrypted
Content originally encrypted but stored decrypted in image.
@ SectorStatusMode2Form2NoCrc
Suffix matches MODE 2 Form 2 but CRC empty/missing.
@ SectorStatusMode1Correct
Valid MODE 1 data with regenerable suffix/prefix.
@ SectorStatusMode2Form2Ok
Suffix matches MODE 2 Form 2 with valid CRC.
@ SectorStatusErrored
Error during dumping; data may be incomplete or corrupt.
@ SectorStatusMode2Form1Ok
Suffix verified/regenerable for MODE 2 Form 1.
@ OpticalDisc
Purely optical discs.
@ BlockMedia
Media that is physically block-based or abstracted like that.
@ kTrackTypeCdMode2Form2
Compact Disc Mode 2 Form 2 data track.
@ kTrackTypeData
Generic data track (not further specified).
@ kTrackTypeCdMode2Form1
Compact Disc Mode 2 Form 1 data track.
@ kTrackTypeAudio
Audio track.
@ kTrackTypeCdMode2Formless
Compact Disc Mode 2 (formless) data track.
@ kTrackTypeCdMode1
Compact Disc Mode 1 data track.
@ kDataTypeUserData
User (main) data.
@ kCompressionLzma
LZMA compression.
@ kCompressionNone
Not compressed.
@ kCompressionZstd
Zstandard compression.
@ kCompressionFlac
FLAC compression.
void ec_accumulate_data_block(aaruformat_context *ctx, const BlockHeader *block_header, const uint8_t *lzma_props, const uint8_t *payload, uint32_t payload_size, uint64_t file_offset)
Accumulate parity for a data block that was just written to disk.
#define AARUF_ERROR_CANNOT_WRITE_BLOCK_DATA
Failure writing block payload.
#define AARUF_STATUS_OK
Sector present and read without uncorrectable errors.
#define AARUF_READ_ONLY
Operation requires write mode but context is read-only.
#define AARUF_ERROR_INCORRECT_MEDIA_TYPE
Operation incompatible with image media type.
#define AARUF_ERROR_TRACK_NOT_FOUND
Referenced track number not present.
#define AARUF_ERROR_NOT_ENOUGH_MEMORY
Memory allocation failure (critical).
#define AARUF_ERROR_SECTOR_OUT_OF_BOUNDS
Requested logical sector outside media bounds.
#define AARUF_ERROR_INCORRECT_DATA_SIZE
Data size does not match expected size.
#define AARUF_ERROR_CANNOT_SET_DDT_ENTRY
Failed to encode/store a DDT entry (overflow or IO).
#define AARUF_ERROR_NOT_AARUFORMAT
Input file/stream failed magic or structural validation.
#define AARUF_ERROR_INVALID_TAG
Invalid or unsupported media or sector tag format.
#define AARUF_ERROR_INVALID_SECTOR_LENGTH
Sector length is too big.
#define AARUF_ERROR_CANNOT_WRITE_BLOCK_HEADER
Failure writing block header.
bool lookup_map(const hash_map_t *map, uint64_t key, uint64_t *out_value)
Looks up a value by key in the hash map.
bool insert_map(hash_map_t *map, uint64_t key, uint64_t value)
Inserts a key-value pair into the hash map.
bool set_ddt_entry_v2(aaruformat_context *ctx, uint64_t sector_address, bool negative, uint64_t offset, uint64_t block_offset, uint8_t sector_status, uint64_t *ddt_entry)
Sets a DDT v2 entry for a given sector address.
static int aaruf_fseek(FILE *stream, aaru_off_t offset, int origin)
#define LZMA_THREADS(ctx)
Clamp num_threads to LZMA's valid range [1, 2].
Structure definitions and conversion/serialization function declarations for Lisa family disk tags.
uint8_t * priam_tag_to_bytes(priam_tag tag)
Serialize a priam_tag into a newly allocated 24-byte big-endian on-disk representation.
priam_tag bytes_to_priam_tag(const uint8_t *bytes)
Parse a 24-byte Priam tag record into a priam_tag structure.
sony_tag profile_tag_to_sony(profile_tag tag)
Convert a profile_tag to a sony_tag.
uint8_t * sony_tag_to_bytes(sony_tag tag)
Serialize a sony_tag into a newly allocated 12-byte big-endian on-disk representation.
profile_tag priam_tag_to_profile(priam_tag tag)
Convert a priam_tag to a profile_tag.
uint8_t * profile_tag_to_bytes(profile_tag tag)
Serialize a profile_tag into a newly allocated 20-byte big-endian on-disk representation.
profile_tag sony_tag_to_profile(sony_tag tag)
Convert a sony_tag to a profile_tag representation.
sony_tag bytes_to_sony_tag(const uint8_t *bytes)
Parse a 12-byte Sony tag record into a sony_tag structure.
profile_tag bytes_to_profile_tag(const uint8_t *bytes)
Parse a 20-byte Profile tag record into a profile_tag structure.
priam_tag sony_tag_to_priam(sony_tag tag)
Convert a sony_tag to a priam_tag representation.
priam_tag profile_tag_to_priam(profile_tag tag)
Convert a profile_tag to a priam_tag.
sony_tag priam_tag_to_sony(priam_tag tag)
Convert a priam_tag to a sony_tag.
void ps3_lazy_init(aaruformat_context *ctx)
Lazy-initialize PS3 encryption state from context media tags.
void ps3_decrypt_sector(const uint8_t disc_key[16], uint64_t sector_num, uint8_t *data, uint32_t length)
Decrypt a sector using PS3 disc encryption (AES-128-CBC).
bool ps3_is_sector_encrypted(const Ps3PlaintextRegion *plaintext_regions, uint32_t region_count, uint64_t sector_address)
Check whether a sector is encrypted (i.e., not in any plaintext region).
uint32_t MediaType
Media type identifier (see MediaType enum; 0=Unknown).
uint8_t MetadataMediaType
Media type for sidecar generation (internal archival use).
uint64_t Sectors
Total count of addressable logical sectors/blocks.
Single index entry describing a block's type, (optional) data classification, and file offset.
uint32_t blockType
Block identifier of the referenced block (value from BlockType).
uint64_t offset
Absolute byte offset in the image where the referenced block header begins.
uint16_t dataType
Data classification (value from DataType) or unused for untyped blocks.
A plaintext (unencrypted) region on a PS3 disc.
Single optical disc track descriptor (sequence, type, LBAs, session, ISRC, flags).
uint8_t session
Session number (1-based). 1 for single-session discs.
uint8_t flags
Control / attribute bitfield (see file documentation for suggested bit mapping).
int64_t end
Inclusive ending LBA of the track.
uint8_t sequence
Track number (1..99 typical for CD audio/data). 0 may indicate placeholder/non-standard.
int64_t start
Inclusive starting LBA of the track.
uint8_t type
Track type (value from TrackType).
uint8_t isrc[13]
ISRC raw 13-byte code (no null terminator). All zeros if not present.
Master context representing an open or in‑creation Aaru image.
uint8_t * ps3_disc_key
Cached disc key (16 bytes), NULL if not loaded.
DdtHeader2 user_data_ddt_header
Active user data DDT v2 header (primary table meta).
bool dirty_checksum_block
True if checksum block should be written during close.
bool deduplicate
Storage deduplication active (duplicates coalesce).
size_t sector_suffix_length
Length of sector_suffix.
bool compression_enabled
True if block compression enabled (writing path).
uint64_t last_written_block
Last written block number (write path).
uint8_t * sector_cpr_mai
DVD sector CPR_MAI (6 bytes) if present.
bool dirty_media_tags
True if media tags should be written during close.
hash_map_t * sector_hash_map
Deduplication hash map (fingerprint->entry mapping).
uint32_t ps3_plaintext_region_count
Number of plaintext regions.
sha256_ctx sha256_context
Opaque SHA-256 context for streaming updates.
bool calculating_sha256
True if whole-image SHA-256 being calculated on-the-fly.
uint8_t * sector_ied
DVD sector IED (2 bytes) if present.
md5_ctx md5_context
Opaque MD5 context for streaming updates.
bool dirty_sector_suffix_block
True if sector suffix block should be written during close.
uint8_t * sector_prefix
Raw per-sector prefix (e.g., sync+header) uncorrected.
bool dirty_dvd_title_key_decrypted_block
True if decrypted title key block should be written during close.
uint64_t * sector_suffix_ddt2
CD sector suffix DDT V2.
bool dirty_mode2_subheaders_block
True if MODE2 subheader block should be written during close.
uint8_t * sector_edc
DVD sector EDC (4 bytes) if present.
bool calculating_sha1
True if whole-image SHA-1 being calculated on-the-fly.
bool dirty_tracks_block
True if tracks block should be written during close.
CdEccContext * ecc_cd_context
CD ECC/EDC helper tables (allocated on demand).
bool rewinded
True if stream has been rewound after open (write path).
bool ps3_encryption_initialized
Whether lazy init has occurred.
bool dirty_sector_prefix_block
True if sector prefix block should be written during close.
bool dirty_index_block
True if index block should be written during close.
uint8_t * sector_suffix
Raw per-sector suffix (EDC/ECC) uncorrected.
AaruHeaderV2 header
Parsed container header (v2).
int current_block_offset
Logical offset inside block (units: bytes or sectors depending on path).
bool is_writing
True if context opened/created for writing.
spamsum_ctx * spamsum_context
Opaque SpamSum context for streaming updates.
size_t sector_prefix_offset
Current position in sector_prefix.
BlockHeader current_block_header
Header for block currently being assembled (write path).
uint64_t magic
File magic (AARU_MAGIC) post-open.
uint8_t * writing_buffer
Accumulation buffer for current block data.
uint64_t * sector_prefix_ddt2
CD sector prefix DDT V2.
bool calculating_spamsum
True if whole-image SpamSum being calculated on-the-fly.
size_t sector_prefix_length
Length of sector_prefix.
mediaTagEntry * mediaTags
Hash table of extra media tags (uthash root).
blake3_hasher * blake3_context
Opaque BLAKE3 context for streaming updates.
bool calculating_blake3
True if whole-image BLAKE3 being calculated on-the-fly.
uint64_t next_block_position
Absolute file offset where next block will be written.
bool calculating_md5
True if whole-image MD5 being calculated on-the-fly.
bool dirty_sector_suffix_ddt
True if sector suffix DDT should be written during close.
size_t sector_suffix_offset
Current position in sector_suffix.
bool has_zstd_blocks
True if any block was actually written with Zstandard compression.
int zstd_level
Zstandard compression level (writing path, default 19).
uint8_t * sector_decrypted_title_key
DVD decrypted title key (5 bytes) if present.
int writing_buffer_position
Current size / position within writingBuffer.
bool block_zero_written
True if block zero has been written (writing path).
crc64_ctx * crc64_context
Opaque CRC64 context for streaming updates.
uint8_t * sector_subchannel
Raw 96-byte subchannel (if captured).
bool dirty_sector_subchannel_block
True if subchannel block should be written during close.
FILE * imageStream
Underlying FILE* stream (binary mode).
bool dirty_dvd_long_sector_blocks
True if DVD long sector blocks should be written during close.
bool dirty_sector_prefix_ddt
True if sector prefix DDT should be written during close.
UT_array * index_entries
Flattened index entries (UT_array of IndexEntry).
int num_threads
Compression worker threads (1 = single-threaded, default).
bool ec_enabled
True if erasure coding is active.
uint8_t * mode2_subheaders
MODE2 Form1/Form2 8-byte subheaders (concatenated).
ImageInfo image_info
Exposed high-level image info summary.
uint8_t * sector_id
DVD sector ID (4 bytes) if present.
void * ps3_plaintext_regions
Parsed Ps3PlaintextRegion array (max 32), NULL if not loaded.
bool use_zstd
Use Zstandard instead of LZMA for data blocks.
sha1_ctx sha1_context
Opaque SHA-1 context for streaming updates.
uint32_t lzma_dict_size
LZMA dictionary size (writing path).
TrackEntry * track_entries
Full track list (tracksHeader.entries elements).
uint8_t current_track_type
Current track type (when writing optical images with tracks, needed for block compression type).
bool writing_long
True if writing long sectors.
TracksHeader tracks_header
Tracks header (optical) if present.
Hash table entry for an arbitrary media tag (e.g., proprietary drive/medium descriptor).
uint8_t * data
Tag data blob (opaque to library core); length bytes long.
int32_t type
Numeric type identifier.
uint32_t length
Length in bytes of data.
int32_t aaruf_close_current_block(aaruformat_context *ctx)
Finalizes and writes the current data block to the AaruFormat image file.
int32_t aaruf_write_sector_tag(void *context, const uint64_t sector_address, const bool negative, const uint8_t *data, const size_t length, const int32_t tag)
Writes per-sector tag data (auxiliary metadata) for a specific sector.
int32_t aaruf_write_media_tag(void *context, const uint8_t *data, const int32_t type, const uint32_t length)
Writes a media tag to the AaruFormat image, storing medium-specific metadata and descriptors.
int32_t aaruf_write_sector(void *context, uint64_t sector_address, bool negative, const uint8_t *data, uint8_t sector_status, uint32_t length)
Writes a sector to the AaruFormat image.
int32_t aaruf_write_sector_long(void *context, uint64_t sector_address, bool negative, const uint8_t *data, uint8_t sector_status, uint32_t length)
Writes a full ("long") raw sector from optical or block media, parsing structure and validating conte...