Major cleanup.
This commit is contained in:
@@ -3,118 +3,180 @@
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
typedef enum {
|
||||
PCM8,
|
||||
PCM16,
|
||||
TODO
|
||||
} CWAVType;
|
||||
enum {
|
||||
CWAV_REF_DSP_ADPCM_INFO = 0x0300,
|
||||
CWAV_REF_IMA_ADPCM_INFO = 0x0301,
|
||||
CWAV_REF_SAMPLE_DATA = 0x1F00,
|
||||
CWAV_REF_INFO_BLOCK = 0x7000,
|
||||
CWAV_REF_DATA_BLOCK = 0x7001,
|
||||
CWAV_REF_CHANNEL_INFO = 0x7100
|
||||
};
|
||||
|
||||
typedef struct {
|
||||
char magic[4] = {'C', 'W', 'A', 'V'};
|
||||
u16 endianess = 0xFEFF;
|
||||
u16 structLength = 0x40;
|
||||
u32 unknown0 = 0;
|
||||
u16 typeId;
|
||||
u16 padding;
|
||||
u32 offset;
|
||||
} CWAVReference;
|
||||
|
||||
typedef struct {
|
||||
CWAVReference ref;
|
||||
u32 size;
|
||||
} CWAVSizedReference;
|
||||
|
||||
typedef struct {
|
||||
u32 count;
|
||||
CWAVReference contents[0]; // Relative to beginning of CWAVReferenceTable.
|
||||
} CWAVReferenceTable;
|
||||
|
||||
#define CWAV_MAGIC "CWAV"
|
||||
|
||||
enum {
|
||||
CWAV_ENDIANNESS_LITTLE = 0xFEFF,
|
||||
CWAV_ENDIANNESS_BIG = 0xFFFE
|
||||
};
|
||||
|
||||
#define CWAV_VERSION 0x02010000
|
||||
|
||||
typedef struct {
|
||||
char magic[4];
|
||||
u16 endianness;
|
||||
u16 headerSize;
|
||||
u32 version;
|
||||
u32 fileSize;
|
||||
u32 numChunks = 2;
|
||||
u32 infoChunkFlags = 0x7000;
|
||||
u32 infoChunkOffset;
|
||||
u32 infoChunkLength;
|
||||
u32 dataChunkFlags = 0x7000;
|
||||
u32 dataChunkOffset;
|
||||
u32 dataChunkLength;
|
||||
u8 reserved[0x14] = {0};
|
||||
u16 numBlocks;
|
||||
u16 reserved;
|
||||
CWAVSizedReference infoBlock; // Relative to start of file.
|
||||
CWAVSizedReference dataBlock; // Relative to start of file.
|
||||
} CWAVHeader;
|
||||
|
||||
#define CWAV_BLOCK_MAGIC_INFO "INFO"
|
||||
#define CWAV_BLOCK_MAGIC_DATA "DATA"
|
||||
|
||||
typedef struct {
|
||||
char magic[4] = {'I', 'N', 'F', 'O'};
|
||||
u32 length;
|
||||
u32 type;
|
||||
char magic[4];
|
||||
u32 size;
|
||||
} CWAVBlockHeader;
|
||||
|
||||
enum {
|
||||
CWAV_ENCODING_PCM8 = 0,
|
||||
CWAV_ENCODING_PCM16 = 1,
|
||||
CWAV_ENCODING_DSP_ADPCM = 2,
|
||||
CWAV_ENCODING_IMA_ADPCM = 3
|
||||
};
|
||||
|
||||
typedef struct {
|
||||
CWAVBlockHeader header;
|
||||
u8 encoding;
|
||||
bool loop;
|
||||
u16 padding;
|
||||
u32 sampleRate;
|
||||
u32 unknown1 = 0;
|
||||
u32 totalSamples;
|
||||
u32 unknown2 = 0;
|
||||
u32 totalChannels;
|
||||
} CWAVInfoHeader;
|
||||
u32 loopStartFrame;
|
||||
u32 loopEndFrame;
|
||||
u32 reserved;
|
||||
CWAVReferenceTable channelInfos;
|
||||
} CWAVInfoBlockHeader;
|
||||
|
||||
typedef struct {
|
||||
u32 flags = 0x7100;
|
||||
u32 offset;
|
||||
} CWAVChannelDataPointer;
|
||||
CWAVReference samples; // Relative to CWAVDataBlock.data
|
||||
CWAVReference adpcmInfo; // Relative to beginning of CWAVChannelInfo.
|
||||
u32 reserved;
|
||||
} CWAVChannelInfo;
|
||||
|
||||
typedef struct {
|
||||
u32 flags = 0x1F00;
|
||||
u32 offset;
|
||||
u32 unknown3 = 0;
|
||||
u32 unknown4 = 0;
|
||||
u32 padding = 0;
|
||||
} CWAVChannelData;
|
||||
u16 coefficients[16];
|
||||
} CWAVDSPADPCMParam;
|
||||
|
||||
typedef struct {
|
||||
char magic[4] = {'D', 'A', 'T', 'A'};
|
||||
u32 length;
|
||||
} CWAVDataHeader;
|
||||
u8 predictorScale;
|
||||
u8 reserved;
|
||||
u16 previousSample;
|
||||
u16 secondPreviousSample;
|
||||
} CWAVDSPADPCMContext;
|
||||
|
||||
u8* cwav_build(CWAV cwav, u32* size) {
|
||||
CWAVHeader header;
|
||||
u32 offset = sizeof(CWAVHeader);
|
||||
typedef struct {
|
||||
CWAVDSPADPCMParam param;
|
||||
CWAVDSPADPCMContext context;
|
||||
CWAVDSPADPCMContext loopContext;
|
||||
u16 padding;
|
||||
} CWAVDSPADPCMInfo;
|
||||
|
||||
header.infoChunkOffset = offset;
|
||||
header.infoChunkLength = (u32) sizeof(CWAVInfoHeader) + (u32) ((sizeof(CWAVChannelDataPointer) + sizeof(CWAVChannelData)) * cwav.channels);
|
||||
offset += header.infoChunkLength;
|
||||
typedef struct {
|
||||
u16 data;
|
||||
u8 tableIndex;
|
||||
u8 padding;
|
||||
} CWAVIMAADPCMContext;
|
||||
|
||||
header.dataChunkOffset = offset;
|
||||
header.dataChunkLength = (u32) sizeof(CWAVDataHeader) + cwav.dataSize;
|
||||
offset += header.dataChunkLength;
|
||||
typedef struct {
|
||||
CWAVIMAADPCMContext context;
|
||||
CWAVIMAADPCMContext loopContext;
|
||||
} CWAVIMAADPCMInfo;
|
||||
|
||||
header.fileSize = offset;
|
||||
typedef struct {
|
||||
CWAVBlockHeader header;
|
||||
u8 data[0];
|
||||
} CWAVDataBlock;
|
||||
|
||||
u8* output = (u8*) malloc(offset);
|
||||
u32 pos = 0;
|
||||
void* cwav_build(u32* size, CWAV cwav) {
|
||||
u32 headerSize = (sizeof(CWAVHeader) + 0x1F) & ~0x1F;
|
||||
u32 infoSize = ((sizeof(CWAVInfoBlockHeader) + (cwav.channels * (sizeof(CWAVReference) + sizeof(CWAVChannelInfo)))) + 0x1F) & ~0x1F;
|
||||
u32 dataSize = ((sizeof(CWAVDataBlock) + 0x1F) & ~0x1F) + cwav.dataSize;
|
||||
|
||||
memcpy(output + pos, &header, sizeof(CWAVHeader));
|
||||
pos += sizeof(CWAVHeader);
|
||||
u32 outputSize = headerSize + infoSize + dataSize;
|
||||
|
||||
u32 bytesPerSample = (u32) cwav.bitsPerSample / 8;
|
||||
|
||||
CWAVInfoHeader infoHeader;
|
||||
infoHeader.length = header.infoChunkLength;
|
||||
infoHeader.type = cwav.bitsPerSample == 16 ? PCM16 : PCM8;
|
||||
infoHeader.sampleRate = cwav.sampleRate;
|
||||
infoHeader.totalSamples = cwav.dataSize / bytesPerSample / cwav.channels;
|
||||
infoHeader.totalChannels = cwav.channels;
|
||||
memcpy(output + pos, &infoHeader, sizeof(CWAVInfoHeader));
|
||||
pos += sizeof(CWAVInfoHeader);
|
||||
|
||||
for(int i = 0; i < cwav.channels; i++) {
|
||||
CWAVChannelDataPointer pointer;
|
||||
pointer.offset = 0x4 + (cwav.channels * (u32) sizeof(CWAVChannelDataPointer)) + (i * (u32) sizeof(CWAVChannelData));
|
||||
memcpy(output + pos, &pointer, sizeof(CWAVChannelDataPointer));
|
||||
pos += sizeof(CWAVChannelDataPointer);
|
||||
void* output = calloc(outputSize, sizeof(u8));
|
||||
if(output == NULL) {
|
||||
printf("ERROR: Could not allocate memory for CWAV data.\n");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
u32 bytesPerChannel = cwav.dataSize / cwav.channels;
|
||||
for(int i = 0; i < cwav.channels; i++) {
|
||||
CWAVChannelData data;
|
||||
data.offset = i * bytesPerChannel;
|
||||
memcpy(output + pos, &data, sizeof(CWAVChannelData));
|
||||
pos += sizeof(CWAVChannelData);
|
||||
CWAVHeader* header = (CWAVHeader*) &((u8*) output)[0];
|
||||
memcpy(header->magic, CWAV_MAGIC, sizeof(header->magic));
|
||||
header->endianness = CWAV_ENDIANNESS_LITTLE;
|
||||
header->headerSize = (u16) headerSize;
|
||||
header->version = CWAV_VERSION;
|
||||
header->fileSize = outputSize;
|
||||
header->numBlocks = 2;
|
||||
header->infoBlock.ref.typeId = CWAV_REF_INFO_BLOCK;
|
||||
header->infoBlock.ref.offset = headerSize;
|
||||
header->infoBlock.size = infoSize;
|
||||
header->dataBlock.ref.typeId = CWAV_REF_DATA_BLOCK;
|
||||
header->dataBlock.ref.offset = headerSize + infoSize;
|
||||
header->dataBlock.size = dataSize;
|
||||
|
||||
CWAVInfoBlockHeader* infoBlockHeader = (CWAVInfoBlockHeader*) &((u8*) output)[headerSize];
|
||||
memcpy(infoBlockHeader->header.magic, CWAV_BLOCK_MAGIC_INFO, sizeof(infoBlockHeader->header.magic));
|
||||
infoBlockHeader->header.size = infoSize;
|
||||
infoBlockHeader->encoding = cwav.bitsPerSample == 16 ? CWAV_ENCODING_PCM16 : CWAV_ENCODING_PCM8;
|
||||
infoBlockHeader->loop = cwav.loop;
|
||||
infoBlockHeader->sampleRate = cwav.sampleRate;
|
||||
infoBlockHeader->loopStartFrame = cwav.loopStartFrame;
|
||||
infoBlockHeader->loopEndFrame = cwav.loopEndFrame != 0 ? cwav.loopEndFrame : cwav.dataSize / cwav.channels / (cwav.bitsPerSample / 8);
|
||||
infoBlockHeader->channelInfos.count = cwav.channels;
|
||||
for(u32 c = 0; c < cwav.channels; c++) {
|
||||
infoBlockHeader->channelInfos.contents[c].typeId = CWAV_REF_CHANNEL_INFO;
|
||||
infoBlockHeader->channelInfos.contents[c].offset = sizeof(CWAVReferenceTable) + (cwav.channels * sizeof(CWAVReference)) + (c * sizeof(CWAVChannelInfo));
|
||||
|
||||
CWAVChannelInfo* info = (CWAVChannelInfo*) &((u8*) output)[headerSize + sizeof(CWAVInfoBlockHeader) + (cwav.channels * sizeof(CWAVReference)) + (c * sizeof(CWAVChannelInfo))];
|
||||
info->samples.typeId = CWAV_REF_SAMPLE_DATA;
|
||||
info->samples.offset = (((sizeof(CWAVDataBlock) + 0x1F) & ~0x1F) - sizeof(CWAVDataBlock)) + (c * (cwav.dataSize / cwav.channels));
|
||||
info->adpcmInfo.typeId = 0;
|
||||
info->adpcmInfo.offset = 0xFFFFFFFF;
|
||||
}
|
||||
|
||||
CWAVDataHeader dataHeader;
|
||||
dataHeader.length = header.dataChunkLength;
|
||||
memcpy(output + pos, &dataHeader, sizeof(CWAVDataHeader));
|
||||
pos += sizeof(CWAVDataHeader);
|
||||
CWAVDataBlock* dataBlock = (CWAVDataBlock*) &((u8*) output)[headerSize + infoSize];
|
||||
memcpy(dataBlock->header.magic, CWAV_BLOCK_MAGIC_DATA, sizeof(dataBlock->header.magic));
|
||||
dataBlock->header.size = dataSize;
|
||||
|
||||
for(int i = 0; i < cwav.dataSize; i += cwav.channels * bytesPerSample) {
|
||||
for(int c = 0; c < cwav.channels; c++) {
|
||||
memcpy(output + pos + (bytesPerChannel * c) + (i / cwav.channels), cwav.data + i + (c * bytesPerSample), bytesPerSample);
|
||||
for(u32 i = 0; i < cwav.dataSize; i += cwav.channels * (cwav.bitsPerSample / 8)) {
|
||||
for(u32 c = 0; c < cwav.channels; c++) {
|
||||
CWAVChannelInfo* info = (CWAVChannelInfo*) &((u8*) output)[headerSize + sizeof(CWAVInfoBlockHeader) + (cwav.channels * sizeof(CWAVReference)) + (c * sizeof(CWAVChannelInfo))];
|
||||
|
||||
memcpy(&dataBlock->data[info->samples.offset + (i / cwav.channels)], &((u8*) cwav.data)[i + (c * (cwav.bitsPerSample / 8))], cwav.bitsPerSample / 8);
|
||||
}
|
||||
}
|
||||
|
||||
pos += cwav.dataSize;
|
||||
|
||||
if(size != NULL) {
|
||||
*size = pos;
|
||||
*size = outputSize;
|
||||
}
|
||||
|
||||
return output;
|
||||
|
||||
Reference in New Issue
Block a user