67695617a1
git-svn-id: svn://svn.rockbox.org/rockbox/trunk@20883 a1c6a512-1295-4272-9138-f99709370657
584 lines
18 KiB
C
584 lines
18 KiB
C
/***************************************************************************
|
|
* __________ __ ___.
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
* \/ \/ \/ \/ \/
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2008 Dave Chapman
|
|
* Copyright (C) 2009 Mohamed Tarek
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
* KIND, either express or implied.
|
|
*
|
|
****************************************************************************/
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdint.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
|
|
#include "rm2wav.h"
|
|
|
|
#define MAX_PATH 260
|
|
#define PKT_HEADER_SIZE 12
|
|
#define RAW_AUDIO_DATA packet_length-PKT_HEADER_SIZE
|
|
|
|
|
|
#if 0
|
|
#define DEBUG
|
|
#define DEBUGF printf
|
|
#else
|
|
#define DEBUGF(...)
|
|
#endif
|
|
|
|
/* ASF codec IDs */
|
|
#define CODEC_ID_WMAV1 0x160
|
|
#define CODEC_ID_WMAV2 0x161
|
|
|
|
static unsigned char wav_header[44]={
|
|
'R','I','F','F',// 0 - ChunkID
|
|
0,0,0,0, // 4 - ChunkSize (filesize-8)
|
|
'W','A','V','E',// 8 - Format
|
|
'f','m','t',' ',// 12 - SubChunkID
|
|
16,0,0,0, // 16 - SubChunk1ID // 16 for PCM
|
|
1,0, // 20 - AudioFormat (1=Uncompressed)
|
|
2,0, // 22 - NumChannels
|
|
0,0,0,0, // 24 - SampleRate in Hz
|
|
0,0,0,0, // 28 - Byte Rate (SampleRate*NumChannels*(BitsPerSample/8)
|
|
4,0, // 32 - BlockAlign (== NumChannels * BitsPerSample/8)
|
|
16,0, // 34 - BitsPerSample
|
|
'd','a','t','a',// 36 - Subchunk2ID
|
|
0,0,0,0 // 40 - Subchunk2Size
|
|
};
|
|
|
|
int open_wav(char* filename) {
|
|
int fd,res;
|
|
|
|
fd=open(filename,O_CREAT|O_WRONLY|O_TRUNC,S_IRUSR|S_IWUSR);
|
|
if (fd >= 0) {
|
|
res = write(fd,wav_header,sizeof(wav_header));
|
|
}
|
|
|
|
return(fd);
|
|
}
|
|
|
|
void close_wav(int fd, RMContext *rmctx) {
|
|
int x,res;
|
|
int filesize;
|
|
int bytes_per_sample = 2;
|
|
int samples_per_frame = rmctx->samples_pf_pc;
|
|
int nb_channels = rmctx->nb_channels;
|
|
int sample_rate = rmctx->sample_rate;
|
|
int nb_frames = rmctx->audio_framesize/rmctx->block_align * rmctx->nb_packets - 2; // first 2 frames have no valid audio; skipped in output
|
|
|
|
filesize= samples_per_frame*bytes_per_sample*nb_frames +44;
|
|
printf("Filesize = %d\n",filesize);
|
|
|
|
// ChunkSize
|
|
x=filesize-8;
|
|
wav_header[4]=(x&0xff);
|
|
wav_header[5]=(x&0xff00)>>8;
|
|
wav_header[6]=(x&0xff0000)>>16;
|
|
wav_header[7]=(x&0xff000000)>>24;
|
|
|
|
// Number of channels
|
|
wav_header[22]=nb_channels;
|
|
|
|
// Samplerate
|
|
wav_header[24]=sample_rate&0xff;
|
|
wav_header[25]=(sample_rate&0xff00)>>8;
|
|
wav_header[26]=(sample_rate&0xff0000)>>16;
|
|
wav_header[27]=(sample_rate&0xff000000)>>24;
|
|
|
|
// ByteRate
|
|
x=sample_rate*bytes_per_sample*nb_channels;
|
|
wav_header[28]=(x&0xff);
|
|
wav_header[29]=(x&0xff00)>>8;
|
|
wav_header[30]=(x&0xff0000)>>16;
|
|
wav_header[31]=(x&0xff000000)>>24;
|
|
|
|
// BlockAlign
|
|
wav_header[32]=rmctx->block_align;//2*rmctx->nb_channels;
|
|
|
|
// Bits per sample
|
|
wav_header[34]=16;
|
|
|
|
// Subchunk2Size
|
|
x=filesize-44;
|
|
wav_header[40]=(x&0xff);
|
|
wav_header[41]=(x&0xff00)>>8;
|
|
wav_header[42]=(x&0xff0000)>>16;
|
|
wav_header[43]=(x&0xff000000)>>24;
|
|
|
|
lseek(fd,0,SEEK_SET);
|
|
res = write(fd,wav_header,sizeof(wav_header));
|
|
close(fd);
|
|
}
|
|
|
|
/* Some Rockbox-like functions (these should be implemented in metadata_common.[ch] */
|
|
struct cook_extradata {
|
|
uint32_t cook_version;
|
|
uint16_t samples_pf_pc; /* samples per frame per channel */
|
|
uint16_t nb_subbands; /* number of subbands in the frequency domain */
|
|
|
|
/* extra 8 bytes for stereo data */
|
|
uint32_t unused;
|
|
uint16_t js_subband_start; /* joint stereo subband start */
|
|
uint16_t js_vlc_bits;
|
|
};
|
|
|
|
static int read_uint8(int fd, uint8_t* buf)
|
|
{
|
|
unsigned char tmp[1];
|
|
int res;
|
|
|
|
res=read(fd, tmp, 1);
|
|
*buf = tmp[0];
|
|
return res;
|
|
}
|
|
|
|
static int read_uint16be(int fd, uint16_t* buf)
|
|
{
|
|
unsigned char tmp[2];
|
|
int res;
|
|
|
|
res=read(fd, tmp, 2);
|
|
*buf = (tmp[0] << 8) | tmp[1];
|
|
return res;
|
|
}
|
|
|
|
static int read_uint32be(int fd, uint32_t* buf)
|
|
{
|
|
unsigned char tmp[4];
|
|
int res;
|
|
|
|
res=read(fd, tmp, 4);
|
|
*buf = (tmp[0] << 24) | (tmp[1] << 16) | (tmp[2] << 8) | tmp[3];
|
|
return res;
|
|
}
|
|
|
|
off_t filesize(int fd)
|
|
{
|
|
struct stat buf;
|
|
|
|
if (fstat(fd,&buf) == -1) {
|
|
return -1;
|
|
} else {
|
|
return buf.st_size;
|
|
}
|
|
}
|
|
|
|
int read_cook_extradata(int fd, RMContext *rmctx) {
|
|
read_uint32be(fd, &rmctx->cook_version);
|
|
read_uint16be(fd, &rmctx->samples_pf_pc);
|
|
read_uint16be(fd, &rmctx->nb_subbands);
|
|
if(rmctx->extradata_size == 16) {
|
|
read_uint32be(fd, &rmctx->unused);
|
|
read_uint16be(fd, &rmctx->js_subband_start);
|
|
read_uint16be(fd, &rmctx->js_vlc_bits);
|
|
}
|
|
return rmctx->extradata_size; /* for 'skipped' */
|
|
}
|
|
|
|
void print_cook_extradata(RMContext *rmctx) {
|
|
|
|
printf(" cook_version = 0x%08x\n", rmctx->cook_version);
|
|
printf(" samples_per_frame_per_channel = %d\n", rmctx->samples_pf_pc);
|
|
printf(" number_of_subbands_in_freq_domain = %d\n", rmctx->nb_subbands);
|
|
if(rmctx->extradata_size == 16) {
|
|
printf(" joint_stereo_subband_start = %d\n",rmctx->js_subband_start);
|
|
printf(" joint_stereo_vlc_bits = %d\n", rmctx->js_vlc_bits);
|
|
}
|
|
}
|
|
|
|
|
|
struct real_object_t
|
|
{
|
|
uint32_t fourcc;
|
|
uint32_t size;
|
|
uint16_t version;
|
|
};
|
|
|
|
#define FOURCC(a,b,c,d) (((a)<<24) | ((b) << 16) | ((c) << 8) | (d))
|
|
|
|
static int real_read_object_header(int fd, struct real_object_t* obj)
|
|
{
|
|
int n;
|
|
|
|
if ((n = read_uint32be(fd, &obj->fourcc)) <= 0) return n;
|
|
if ((n = read_uint32be(fd, &obj->size)) <= 0) return n;
|
|
if ((n = read_uint16be(fd, &obj->version)) <= 0) return n;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static char* fourcc2str(uint32_t f)
|
|
{
|
|
static char res[5];
|
|
|
|
res[0] = (f & 0xff000000) >> 24;
|
|
res[1] = (f & 0xff0000) >> 16;
|
|
res[2] = (f & 0xff00) >> 8;
|
|
res[3] = (f & 0xff);
|
|
res[4] = 0;
|
|
|
|
return res;
|
|
}
|
|
|
|
static int read_str(int fd, char* buf)
|
|
{
|
|
uint8_t len;
|
|
int res;
|
|
|
|
res = read(fd, &len, 1);
|
|
res = read(fd, buf, len);
|
|
buf[len]=0;
|
|
|
|
return len+1;
|
|
}
|
|
|
|
static int real_read_audio_stream_info(int fd, RMContext *rmctx)
|
|
{
|
|
int skipped = 0;
|
|
uint32_t version;
|
|
struct real_object_t obj;
|
|
memset(&obj,0,sizeof(obj));
|
|
uint32_t header_size;
|
|
uint16_t flavor;
|
|
uint32_t coded_framesize;
|
|
uint32_t unknown1;
|
|
uint32_t unknown2;
|
|
uint32_t unknown3;
|
|
uint16_t unknown4;
|
|
uint16_t unknown5;
|
|
uint16_t unknown6;
|
|
uint16_t unknown7;
|
|
uint32_t unknown8;
|
|
uint8_t interleaver_id_length;
|
|
uint32_t interleaver_id;
|
|
uint8_t fourcc_length;
|
|
uint32_t fourcc = 0;
|
|
uint8_t unknown9;
|
|
uint16_t unknown10;
|
|
uint8_t unknown11;
|
|
|
|
read_uint32be(fd, &version);
|
|
skipped += 4;
|
|
|
|
printf(" version=0x%04x\n",((version >> 16) & 0xff));
|
|
if (((version >> 16) & 0xff) == 3) {
|
|
/* Very old version */
|
|
} else {
|
|
real_read_object_header(fd, &obj);
|
|
skipped += 10;
|
|
read_uint32be(fd, &header_size);
|
|
skipped += 4;
|
|
/* obj.size will be filled with an unknown value, replaced with header_size */
|
|
printf(" Object: %s, size: %d bytes, version: 0x%04x\n",fourcc2str(obj.fourcc),header_size,obj.version);
|
|
|
|
read_uint16be(fd, &flavor);
|
|
read_uint32be(fd, &coded_framesize);
|
|
read_uint32be(fd, &unknown1);
|
|
read_uint32be(fd, &unknown2);
|
|
read_uint32be(fd, &unknown3);
|
|
read_uint16be(fd, &rmctx->sub_packet_h);
|
|
read_uint16be(fd, &rmctx->block_align);
|
|
read_uint16be(fd, &rmctx->sub_packet_size);
|
|
read_uint16be(fd, &unknown4);
|
|
skipped += 26;
|
|
if (((version >> 16) & 0xff) == 5)
|
|
{
|
|
read_uint16be(fd, &unknown5);
|
|
read_uint16be(fd, &unknown6);
|
|
read_uint16be(fd, &unknown7);
|
|
skipped += 6;
|
|
}
|
|
read_uint16be(fd, &rmctx->sample_rate);
|
|
read_uint32be(fd, &unknown8);
|
|
read_uint16be(fd, &rmctx->nb_channels);
|
|
skipped += 8;
|
|
if (((version >> 16) & 0xff) == 4)
|
|
{
|
|
read_uint8(fd, &interleaver_id_length);
|
|
read_uint32be(fd, &interleaver_id);
|
|
read_uint8(fd, &fourcc_length);
|
|
read_uint32be(fd, &fourcc);
|
|
skipped += 10;
|
|
}
|
|
if (((version >> 16) & 0xff) == 5)
|
|
{
|
|
read_uint32be(fd, &interleaver_id);
|
|
read_uint32be(fd, &fourcc);
|
|
skipped += 8;
|
|
}
|
|
read_uint8(fd,&unknown9);
|
|
read_uint16be(fd,&unknown10);
|
|
skipped += 3;
|
|
if (((version >> 16) & 0xff) == 5)
|
|
{
|
|
read_uint8(fd, &unknown11);
|
|
skipped += 1;
|
|
}
|
|
|
|
read_uint32be(fd, &rmctx->extradata_size);
|
|
skipped += 4;
|
|
if(!strncmp(fourcc2str(fourcc),"cook",4))
|
|
skipped += read_cook_extradata(fd, rmctx);
|
|
|
|
|
|
printf(" flavor = %d\n",flavor);
|
|
printf(" coded_frame_size = %d\n",coded_framesize);
|
|
printf(" sub_packet_h = %d\n",rmctx->sub_packet_h);
|
|
printf(" frame_size = %d\n",rmctx->block_align);
|
|
printf(" sub_packet_size = %d\n",rmctx->sub_packet_size);
|
|
printf(" sample_rate= %d\n",rmctx->sample_rate);
|
|
printf(" channels= %d\n",rmctx->nb_channels);
|
|
printf(" fourcc = %s\n",fourcc2str(fourcc));
|
|
printf(" codec_extra_data_length = %d\n",rmctx->extradata_size);
|
|
printf(" codec_extradata :\n");
|
|
print_cook_extradata(rmctx);
|
|
|
|
}
|
|
|
|
return skipped;
|
|
}
|
|
|
|
int real_parse_header(int fd, RMContext *rmctx)
|
|
{
|
|
struct real_object_t obj;
|
|
memset(&obj,0,sizeof(obj));
|
|
int res;
|
|
int skipped;
|
|
off_t curpos;
|
|
|
|
uint32_t unknown1;
|
|
uint32_t unknown2;
|
|
|
|
uint32_t max_bitrate;
|
|
uint32_t avg_bitrate = 0;
|
|
uint32_t max_packet_size;
|
|
uint32_t avg_packet_size;
|
|
uint32_t packet_count;
|
|
uint32_t duration;
|
|
uint32_t preroll;
|
|
uint32_t index_offset;
|
|
uint32_t data_offset;
|
|
uint16_t num_streams;
|
|
uint16_t flags = 0;
|
|
|
|
uint16_t stream_id;
|
|
uint32_t start_time;
|
|
char desc[256];
|
|
char mimetype[256];
|
|
uint32_t codec_data_size;
|
|
uint32_t v;
|
|
|
|
char title[256];
|
|
char author[256];
|
|
char copyright[256];
|
|
char comment[256];
|
|
|
|
uint32_t next_data_off;
|
|
uint8_t header_end;
|
|
|
|
curpos = lseek(fd, 0, SEEK_SET);
|
|
res = real_read_object_header(fd, &obj);
|
|
|
|
if (obj.fourcc == FOURCC('.','r','a',0xfd))
|
|
{
|
|
/* Very old .ra format - not yet supported */
|
|
return -1;
|
|
}
|
|
else if (obj.fourcc != FOURCC('.','R','M','F'))
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
read_uint32be(fd, &unknown1);
|
|
read_uint32be(fd, &unknown2);
|
|
|
|
printf("Object: %s, size: %d bytes, version: 0x%04x, pos: %d\n",fourcc2str(obj.fourcc),(int)obj.size,obj.version,(int)curpos);
|
|
printf(" unknown1=%d (0x%08x)\n",unknown1,unknown1);
|
|
printf(" unknown2=%d (0x%08x)\n",unknown2,unknown2);
|
|
|
|
res = real_read_object_header(fd, &obj);
|
|
header_end = 0;
|
|
while(res)
|
|
{
|
|
printf("Object: %s, size: %d bytes, version: 0x%04x, pos: %d\n",fourcc2str(obj.fourcc),(int)obj.size,obj.version,(int)curpos);
|
|
skipped = 10;
|
|
if(obj.fourcc == FOURCC('I','N','D','X'))
|
|
break;
|
|
switch (obj.fourcc)
|
|
{
|
|
case FOURCC('P','R','O','P'): /* File properties */
|
|
read_uint32be(fd, &max_bitrate);
|
|
read_uint32be(fd, &rmctx->bit_rate); /*avg bitrate*/
|
|
read_uint32be(fd, &max_packet_size);
|
|
read_uint32be(fd, &avg_packet_size);
|
|
read_uint32be(fd, &packet_count);
|
|
read_uint32be(fd, &duration);
|
|
read_uint32be(fd, &preroll);
|
|
read_uint32be(fd, &index_offset);
|
|
read_uint32be(fd, &data_offset);
|
|
read_uint16be(fd, &num_streams);
|
|
read_uint16be(fd, &rmctx->flags);
|
|
skipped += 40;
|
|
|
|
printf(" max_bitrate = %d\n",max_bitrate);
|
|
printf(" avg_bitrate = %d\n",avg_bitrate);
|
|
printf(" max_packet_size = %d\n",max_packet_size);
|
|
printf(" avg_packet_size = %d\n",avg_packet_size);
|
|
printf(" packet_count = %d\n",packet_count);
|
|
printf(" duration = %d\n",duration);
|
|
printf(" preroll = %d\n",preroll);
|
|
printf(" index_offset = %d\n",index_offset);
|
|
printf(" data_offset = %d\n",data_offset);
|
|
printf(" num_streams = %d\n",num_streams);
|
|
printf(" flags=0x%04x\n",flags);
|
|
break;
|
|
|
|
case FOURCC('C','O','N','T'):
|
|
/* Four strings - Title, Author, Copyright, Comment */
|
|
skipped += read_str(fd,title);
|
|
skipped += read_str(fd,author);
|
|
skipped += read_str(fd,copyright);
|
|
skipped += read_str(fd,comment);
|
|
|
|
printf(" title=\"%s\"\n",title);
|
|
printf(" author=\"%s\"\n",author);
|
|
printf(" copyright=\"%s\"\n",copyright);
|
|
printf(" comment=\"%s\"\n",comment);
|
|
break;
|
|
|
|
case FOURCC('M','D','P','R'): /* Media properties */
|
|
read_uint16be(fd,&stream_id);
|
|
skipped += 2;
|
|
read_uint32be(fd,&max_bitrate);
|
|
skipped += 4;
|
|
read_uint32be(fd,&avg_bitrate);
|
|
skipped += 4;
|
|
read_uint32be(fd,&max_packet_size);
|
|
skipped += 4;
|
|
read_uint32be(fd,&avg_packet_size);
|
|
skipped += 4;
|
|
read_uint32be(fd,&start_time);
|
|
skipped += 4;
|
|
read_uint32be(fd,&preroll);
|
|
skipped += 4;
|
|
read_uint32be(fd,&duration);
|
|
skipped += 4;
|
|
skipped += read_str(fd,desc);
|
|
skipped += read_str(fd,mimetype);
|
|
read_uint32be(fd,&codec_data_size);
|
|
skipped += 4;
|
|
//From ffmpeg: codec_pos = url_ftell(pb);
|
|
read_uint32be(fd,&v);
|
|
skipped += 4;
|
|
|
|
printf(" stream_id = 0x%04x\n",stream_id);
|
|
printf(" max_bitrate = %d\n",max_bitrate);
|
|
printf(" avg_bitrate = %d\n",avg_bitrate);
|
|
printf(" max_packet_size = %d\n",max_packet_size);
|
|
printf(" avg_packet_size = %d\n",avg_packet_size);
|
|
printf(" start_time = %d\n",start_time);
|
|
printf(" preroll = %d\n",preroll);
|
|
printf(" duration = %d\n",duration);
|
|
printf(" desc=\"%s\"\n",desc);
|
|
printf(" mimetype=\"%s\"\n",mimetype);
|
|
printf(" codec_data_size = %d\n",codec_data_size);
|
|
printf(" v=\"%s\"\n", fourcc2str(v));
|
|
|
|
if (v == FOURCC('.','r','a',0xfd))
|
|
{
|
|
skipped += real_read_audio_stream_info(fd, rmctx);
|
|
}
|
|
|
|
break;
|
|
|
|
case FOURCC('D','A','T','A'):
|
|
|
|
read_uint32be(fd,&rmctx->nb_packets);
|
|
skipped += 4;
|
|
read_uint32be(fd,&next_data_off);
|
|
skipped += 4;
|
|
if (!rmctx->nb_packets && (rmctx->flags & 4))
|
|
rmctx->nb_packets = 3600 * 25;
|
|
printf(" data_nb_packets = %d\n",rmctx->nb_packets);
|
|
printf(" next DATA offset = %d\n",next_data_off);
|
|
header_end = 1;
|
|
break;
|
|
}
|
|
if(header_end) break;
|
|
curpos = lseek(fd, obj.size - skipped, SEEK_CUR);
|
|
res = real_read_object_header(fd, &obj);
|
|
}
|
|
|
|
|
|
return 0;
|
|
}
|
|
|
|
void rm_get_packet(int fd,RMContext *rmctx, RMPacket *pkt)
|
|
{
|
|
uint8_t unknown,packet_group;
|
|
uint16_t x, place;
|
|
uint16_t sps = rmctx->sub_packet_size;
|
|
uint16_t h = rmctx->sub_packet_h;
|
|
uint16_t y = rmctx->sub_packet_cnt;
|
|
uint16_t w = rmctx->audio_framesize;
|
|
do
|
|
{
|
|
y = rmctx->sub_packet_cnt;
|
|
read_uint16be(fd,&pkt->version);
|
|
read_uint16be(fd,&pkt->length);
|
|
read_uint16be(fd,&pkt->stream_number);
|
|
read_uint32be(fd,&pkt->timestamp);
|
|
DEBUGF(" version = %d\n"
|
|
" length = %d\n"
|
|
" stream = %d\n"
|
|
" timestmp= %d\n",pkt->version,pkt->length,pkt->stream_number,pkt->timestamp);
|
|
|
|
//getchar();
|
|
if(pkt->version == 0)
|
|
{
|
|
read_uint8(fd,&packet_group);
|
|
read_uint8(fd,&pkt->flags);
|
|
}
|
|
if(pkt->version == 1)
|
|
read_uint8(fd,&unknown);
|
|
|
|
if (pkt->flags & 2) /* keyframe */
|
|
y = rmctx->sub_packet_cnt = 0;
|
|
if (!y) /* if keyframe update playback elapsed time */
|
|
rmctx->audiotimestamp = pkt->timestamp;
|
|
|
|
for(x = 0 ; x < w/sps; x++)
|
|
{
|
|
place = sps*(h*x+((h+1)/2)*(y&1)+(y>>1));
|
|
read(fd,pkt->data+place, sps);
|
|
//DEBUGF("place = %d data[place] = %d\n",place,pkt->data[place]);
|
|
}
|
|
rmctx->audio_pkt_cnt++;
|
|
}while(++(rmctx->sub_packet_cnt) < h);
|
|
|
|
//return pkt->data;
|
|
}
|
|
#ifdef DEBUG
|
|
void dump_rm_context(RMContext *rmctx)
|
|
{
|
|
DEBUGF("block_align = %d\n", rmctx->block_align);
|
|
DEBUGF("nb_channels = %d\n", rmctx->nb_channels);
|
|
DEBUGF("sample_rate = %d\n", rmctx->sample_rate);
|
|
DEBUGF("bit_rate = %d\n", rmctx->bit_rate );
|
|
}
|
|
#endif
|