mirror of
https://github.com/aaru-dps/Aaru.git
synced 2025-12-16 19:24:25 +00:00
492 lines
18 KiB
C#
492 lines
18 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Runtime.InteropServices;
|
|
using System.Text;
|
|
using Aaru.CommonTypes;
|
|
using Aaru.CommonTypes.Enums;
|
|
using Aaru.Helpers;
|
|
using Aaru.Logging;
|
|
using Marshal = System.Runtime.InteropServices.Marshal;
|
|
|
|
namespace Aaru.Images;
|
|
|
|
public sealed partial class AaruFormat
|
|
{
|
|
// AARU_EXPORT int32_t AARU_CALL aaruf_write_sector(void *context, uint64_t sector_address, bool negative,
|
|
// const uint8_t *data, uint8_t sector_status, uint32_t length)
|
|
[LibraryImport("libaaruformat", EntryPoint = "aaruf_write_sector", SetLastError = true)]
|
|
[UnmanagedCallConv(CallConvs = [typeof(CallConvStdcall)])]
|
|
private static partial Status aaruf_write_sector(IntPtr context, ulong sectorAddress,
|
|
[MarshalAs(UnmanagedType.I4)] bool negative, [In] byte[] data,
|
|
SectorStatus sectorStatus, uint length);
|
|
|
|
// AARU_EXPORT int32_t AARU_CALL aaruf_write_sector_long(void *context, uint64_t sector_address, bool negative,
|
|
// const uint8_t *data, uint8_t sector_status, uint32_t length)
|
|
[LibraryImport("libaaruformat", EntryPoint = "aaruf_write_sector_long", SetLastError = true)]
|
|
[UnmanagedCallConv(CallConvs = [typeof(CallConvStdcall)])]
|
|
private static partial Status aaruf_write_sector_long(IntPtr context, ulong sectorAddress,
|
|
[MarshalAs(UnmanagedType.I4)] bool negative, [In] byte[] data,
|
|
SectorStatus sectorStatus, uint length);
|
|
|
|
// AARU_EXPORT int32_t AARU_CALL aaruf_write_media_tag(void *context, const uint8_t *data, const int32_t type,
|
|
// const uint32_t length)
|
|
[LibraryImport("libaaruformat", EntryPoint = "aaruf_write_media_tag", SetLastError = true)]
|
|
[UnmanagedCallConv(CallConvs = [typeof(CallConvStdcall)])]
|
|
private static partial Status aaruf_write_media_tag(IntPtr context, [In] byte[] data, MediaTagType type,
|
|
uint length);
|
|
|
|
// AARU_EXPORT int32_t AARU_CALL 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)
|
|
[LibraryImport("libaaruformat", EntryPoint = "aaruf_write_sector_tag", SetLastError = true)]
|
|
[UnmanagedCallConv(CallConvs = [typeof(CallConvStdcall)])]
|
|
private static partial Status aaruf_write_sector_tag(IntPtr context, ulong sectorAddress,
|
|
[MarshalAs(UnmanagedType.I4)] bool negative, [In] byte[] data,
|
|
nuint length, SectorTagType tag);
|
|
|
|
// AARU_EXPORT void AARU_CALL *aaruf_create(const char *filepath, const uint32_t media_type, const uint32_t sector_size,
|
|
// const uint64_t user_sectors, const uint64_t negative_sectors,
|
|
// const uint64_t overflow_sectors, const char *options,
|
|
// const uint8_t *application_name, const uint8_t application_name_length,
|
|
// const uint8_t application_major_version,
|
|
// const uint8_t application_minor_version, const bool is_tape)
|
|
[LibraryImport("libaaruformat",
|
|
EntryPoint = "aaruf_create",
|
|
SetLastError = true,
|
|
StringMarshalling = StringMarshalling.Utf8)]
|
|
[UnmanagedCallConv(CallConvs = [typeof(CallConvStdcall)])]
|
|
private static partial IntPtr aaruf_create(string filepath, MediaType mediaType, uint sectorSize, ulong userSectors,
|
|
ulong negativeSectors, ulong overflowSectors, string options,
|
|
string applicationName, byte applicationNameLength,
|
|
byte applicationMajorVersion, byte applicationMinorVersion,
|
|
[MarshalAs(UnmanagedType.I4)] bool isTape);
|
|
|
|
#region IWritableOpticalImage Members
|
|
|
|
/// <inheritdoc />
|
|
public bool WriteSector(byte[] data, ulong sectorAddress, bool negative, SectorStatus sectorStatus)
|
|
{
|
|
Status res = aaruf_write_sector(_context, sectorAddress, negative, data, sectorStatus, (uint)data.Length);
|
|
|
|
if(res == Status.Ok) return true;
|
|
|
|
ErrorMessage = StatusToErrorMessage(res);
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public bool WriteSectorLong(byte[] data, ulong sectorAddress, bool negative, SectorStatus sectorStatus)
|
|
{
|
|
Status res = aaruf_write_sector_long(_context, sectorAddress, negative, data, sectorStatus, (uint)data.Length);
|
|
|
|
if(res == Status.Ok) return true;
|
|
|
|
ErrorMessage = StatusToErrorMessage(res);
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public bool WriteMediaTag(byte[] data, MediaTagType tag)
|
|
{
|
|
Status res = aaruf_write_media_tag(_context, data, tag, (uint)data.Length);
|
|
|
|
if(res == Status.Ok) return true;
|
|
|
|
ErrorMessage = StatusToErrorMessage(res);
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public bool WriteSectorTag(byte[] data, ulong sectorAddress, bool negative, SectorTagType tag)
|
|
{
|
|
switch(tag)
|
|
{
|
|
// Due to how we cache the tracks, we need to handle ISRC tags ourselves
|
|
case SectorTagType.CdTrackIsrc:
|
|
_trackIsrcs ??= [];
|
|
_trackIsrcs[(int)sectorAddress] = Encoding.UTF8.GetString(data).TrimEnd('\0');
|
|
|
|
return true;
|
|
|
|
// Due to how we cache the tracks, we need to handle Track Flags ourselves
|
|
case SectorTagType.CdTrackFlags:
|
|
{
|
|
_trackFlags ??= [];
|
|
|
|
if(data.Length <= 0) return false;
|
|
|
|
_trackFlags[(int)sectorAddress] = data[0];
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
Status res = aaruf_write_sector_tag(_context, sectorAddress, negative, data, (nuint)data.Length, tag);
|
|
|
|
if(res == Status.Ok) return true;
|
|
|
|
ErrorMessage = StatusToErrorMessage(res);
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public bool WriteSectors(byte[] data, ulong sectorAddress, bool negative, uint length, SectorStatus[] sectorStatus)
|
|
{
|
|
var sectorSize = (uint)(data.Length / length);
|
|
|
|
for(uint i = 0; i < length; i++)
|
|
{
|
|
var sectorData = new byte[sectorSize];
|
|
Array.Copy(data, i * sectorSize, sectorData, 0, sectorSize);
|
|
|
|
if(!WriteSector(sectorData, sectorAddress + i, negative, sectorStatus[i])) return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public bool WriteSectorsLong(byte[] data, ulong sectorAddress, bool negative, uint length,
|
|
SectorStatus[] sectorStatus)
|
|
{
|
|
var sectorSize = (uint)(data.Length / length);
|
|
|
|
for(uint i = 0; i < length; i++)
|
|
{
|
|
var sectorData = new byte[sectorSize];
|
|
Array.Copy(data, i * sectorSize, sectorData, 0, sectorSize);
|
|
|
|
if(!WriteSectorLong(sectorData, sectorAddress + i, negative, sectorStatus[i])) return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public bool WriteSectorsTag(byte[] data, ulong sectorAddress, bool negative, uint length, SectorTagType tag)
|
|
{
|
|
var sectorSize = (uint)(data.Length / length);
|
|
|
|
for(uint i = 0; i < length; i++)
|
|
{
|
|
var sectorData = new byte[sectorSize];
|
|
Array.Copy(data, i * sectorSize, sectorData, 0, sectorSize);
|
|
|
|
if(!WriteSectorTag(sectorData, sectorAddress + i, negative, tag)) return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <inheritdoc />
|
|
public bool Create(string path, MediaType mediaType, Dictionary<string, string> options, ulong sectors,
|
|
uint negativeSectors, uint overflowSectors, uint sectorSize)
|
|
{
|
|
// Convert options dictionary to string format
|
|
string optionsString = options is { Count: > 0 }
|
|
? string.Join(";", options.Select(static kvp => $"{kvp.Key}={kvp.Value}"))
|
|
: null;
|
|
|
|
// Create new image
|
|
if(!File.Exists(path))
|
|
{
|
|
// Get application major and minor version
|
|
Version version = typeof(AaruFormat).Assembly.GetName().Version;
|
|
var majorVersion = (byte)(version?.Major ?? 0);
|
|
var minorVersion = (byte)(version?.Minor ?? 0);
|
|
|
|
|
|
const string applicationName = "Aaru";
|
|
|
|
// Create the image context
|
|
_context = aaruf_create(path,
|
|
mediaType,
|
|
sectorSize,
|
|
sectors,
|
|
negativeSectors,
|
|
overflowSectors,
|
|
optionsString,
|
|
applicationName,
|
|
(byte)applicationName.Length,
|
|
majorVersion,
|
|
minorVersion,
|
|
IsTape);
|
|
|
|
if(_context != IntPtr.Zero) return true;
|
|
|
|
int errno = Marshal.GetLastWin32Error();
|
|
var status = (Status)errno;
|
|
ErrorMessage = StatusToErrorMessage(status);
|
|
|
|
return false;
|
|
}
|
|
|
|
_context = aaruf_open(path, true, optionsString);
|
|
|
|
if(_context == IntPtr.Zero)
|
|
{
|
|
int errno = Marshal.GetLastWin32Error();
|
|
|
|
AaruLogging.Debug(MODULE_NAME,
|
|
"Failed to open AaruFormat image {0}, libaaruformat returned error number {1}",
|
|
path,
|
|
errno);
|
|
|
|
return false;
|
|
}
|
|
|
|
AaruFormatImageInfo imageInfo = new();
|
|
|
|
Status ret = aaruf_get_image_info(_context, ref imageInfo);
|
|
|
|
if(ret != Status.Ok)
|
|
{
|
|
ErrorMessage = StatusToErrorMessage(ret);
|
|
|
|
return false;
|
|
}
|
|
|
|
_imageInfo.Application = StringHandlers.CToString(imageInfo.Application, Encoding.UTF8);
|
|
_imageInfo.Version = StringHandlers.CToString(imageInfo.Version, Encoding.UTF8);
|
|
_imageInfo.ApplicationVersion = StringHandlers.CToString(imageInfo.ApplicationVersion, Encoding.UTF8);
|
|
_imageInfo.CreationTime = DateTime.FromFileTimeUtc(imageInfo.CreationTime);
|
|
_imageInfo.HasPartitions = imageInfo.HasPartitions;
|
|
_imageInfo.HasSessions = imageInfo.HasSessions;
|
|
_imageInfo.ImageSize = imageInfo.ImageSize;
|
|
_imageInfo.MediaType = imageInfo.MediaType;
|
|
_imageInfo.LastModificationTime = DateTime.FromFileTimeUtc(imageInfo.LastModificationTime);
|
|
_imageInfo.SectorSize = imageInfo.SectorSize;
|
|
_imageInfo.Sectors = imageInfo.Sectors;
|
|
_imageInfo.MetadataMediaType = imageInfo.MetadataMediaType;
|
|
|
|
nuint sizet_length = 0;
|
|
|
|
ret = aaruf_get_readable_sector_tags(_context, null, ref sizet_length);
|
|
|
|
if(ret != Status.BufferTooSmall)
|
|
{
|
|
ErrorMessage = StatusToErrorMessage(ret);
|
|
|
|
return false;
|
|
}
|
|
|
|
var sectorTagsBuffer = new byte[sizet_length];
|
|
ret = aaruf_get_readable_sector_tags(_context, sectorTagsBuffer, ref sizet_length);
|
|
|
|
if(ret != Status.Ok)
|
|
{
|
|
ErrorMessage = StatusToErrorMessage(ret);
|
|
|
|
return false;
|
|
}
|
|
|
|
// Convert array of booleans to List of enums
|
|
for(nuint i = 0; i < sizet_length; i++)
|
|
if(sectorTagsBuffer[i] != 0)
|
|
_imageInfo.ReadableSectorTags.Add((SectorTagType)i);
|
|
|
|
sizet_length = 0;
|
|
ret = aaruf_get_readable_media_tags(_context, null, ref sizet_length);
|
|
|
|
if(ret != Status.BufferTooSmall)
|
|
{
|
|
ErrorMessage = StatusToErrorMessage(ret);
|
|
|
|
return false;
|
|
}
|
|
|
|
var mediaTagsBuffer = new byte[sizet_length];
|
|
ret = aaruf_get_readable_media_tags(_context, mediaTagsBuffer, ref sizet_length);
|
|
|
|
if(ret != Status.Ok)
|
|
{
|
|
ErrorMessage = StatusToErrorMessage(ret);
|
|
|
|
return false;
|
|
}
|
|
|
|
// Convert array of booleans to List of enums
|
|
for(nuint i = 0; i < sizet_length; i++)
|
|
if(mediaTagsBuffer[i] != 0)
|
|
_imageInfo.ReadableMediaTags.Add((MediaTagType)i);
|
|
|
|
ret = aaruf_get_media_sequence(_context, out int sequence, out int lastSequence);
|
|
|
|
if(ret == Status.Ok)
|
|
{
|
|
_imageInfo.LastMediaSequence = lastSequence;
|
|
_imageInfo.MediaSequence = sequence;
|
|
}
|
|
|
|
var length = 0;
|
|
ret = aaruf_get_creator(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_creator(_context, buffer, ref length);
|
|
if(ret == Status.Ok) _imageInfo.Creator = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_creator(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_creator(_context, buffer, ref length);
|
|
if(ret == Status.Ok) _imageInfo.Creator = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_comments(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_comments(_context, buffer, ref length);
|
|
if(ret == Status.Ok) _imageInfo.Comments = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_media_title(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_media_title(_context, buffer, ref length);
|
|
if(ret == Status.Ok) _imageInfo.MediaTitle = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_media_manufacturer(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_media_manufacturer(_context, buffer, ref length);
|
|
|
|
if(ret == Status.Ok)
|
|
_imageInfo.MediaManufacturer = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_media_model(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_media_model(_context, buffer, ref length);
|
|
if(ret == Status.Ok) _imageInfo.MediaModel = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_media_serial_number(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_media_serial_number(_context, buffer, ref length);
|
|
|
|
if(ret == Status.Ok)
|
|
_imageInfo.MediaSerialNumber = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_media_barcode(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_media_barcode(_context, buffer, ref length);
|
|
if(ret == Status.Ok) _imageInfo.MediaBarcode = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_media_part_number(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_media_part_number(_context, buffer, ref length);
|
|
if(ret == Status.Ok) _imageInfo.MediaPartNumber = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_drive_manufacturer(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_drive_manufacturer(_context, buffer, ref length);
|
|
|
|
if(ret == Status.Ok)
|
|
_imageInfo.DriveManufacturer = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_drive_model(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_drive_model(_context, buffer, ref length);
|
|
if(ret == Status.Ok) _imageInfo.DriveModel = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_drive_serial_number(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_drive_serial_number(_context, buffer, ref length);
|
|
|
|
if(ret == Status.Ok)
|
|
_imageInfo.DriveSerialNumber = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
length = 0;
|
|
ret = aaruf_get_drive_firmware_revision(_context, null, ref length);
|
|
|
|
if(ret == Status.BufferTooSmall)
|
|
{
|
|
var buffer = new byte[length];
|
|
ret = aaruf_get_drive_firmware_revision(_context, buffer, ref length);
|
|
|
|
if(ret == Status.Ok)
|
|
_imageInfo.DriveFirmwareRevision = StringHandlers.CToString(buffer, Encoding.Unicode, true);
|
|
}
|
|
|
|
ret = aaruf_get_geometry(_context, out uint cylinders, out uint heads, out uint sectorsPerTrack);
|
|
|
|
if(ret == Status.Ok)
|
|
{
|
|
_imageInfo.Cylinders = cylinders;
|
|
_imageInfo.Heads = heads;
|
|
_imageInfo.SectorsPerTrack = sectorsPerTrack;
|
|
}
|
|
|
|
SetMetadataFromTags();
|
|
|
|
ushort infoNegativeSectors = 0;
|
|
|
|
ret = aaruf_get_negative_sectors(_context, ref infoNegativeSectors);
|
|
if(ret == Status.Ok) _imageInfo.NegativeSectors = infoNegativeSectors;
|
|
|
|
ushort infoOverflowSectors = 0;
|
|
|
|
ret = aaruf_get_overflow_sectors(_context, ref infoOverflowSectors);
|
|
if(ret == Status.Ok) _imageInfo.OverflowSectors = infoOverflowSectors;
|
|
|
|
return true;
|
|
}
|
|
|
|
#endregion
|
|
} |