2003-04-30 19:54:54 +00:00
|
|
|
/*
|
|
|
|
Realaudio demuxer for MPlayer
|
2005-03-03 21:13:30 +00:00
|
|
|
(c) 2003, 2005 Roberto Togni
|
2003-04-30 19:54:54 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
#include "mp_msg.h"
|
|
|
|
#include "help_mp.h"
|
|
|
|
|
|
|
|
#include "stream.h"
|
|
|
|
#include "demuxer.h"
|
|
|
|
#include "stheader.h"
|
|
|
|
#include "bswap.h"
|
|
|
|
|
|
|
|
|
|
|
|
#define FOURCC_DOTRA mmioFOURCC('.','r','a', 0xfd)
|
|
|
|
#define FOURCC_144 mmioFOURCC('1','4','_','4')
|
|
|
|
#define FOURCC_288 mmioFOURCC('2','8','_','8')
|
|
|
|
#define FOURCC_DNET mmioFOURCC('d','n','e','t')
|
|
|
|
#define FOURCC_LPCJ mmioFOURCC('l','p','c','J')
|
2006-01-15 21:58:10 +00:00
|
|
|
#define FOURCC_SIPR mmioFOURCC('s','i','p','r')
|
|
|
|
|
|
|
|
|
|
|
|
static unsigned char sipr_swaps[38][2]={
|
|
|
|
{0,63},{1,22},{2,44},{3,90},{5,81},{7,31},{8,86},{9,58},{10,36},{12,68},
|
|
|
|
{13,39},{14,73},{15,53},{16,69},{17,57},{19,88},{20,34},{21,71},{24,46},
|
|
|
|
{25,94},{26,54},{28,75},{29,50},{32,70},{33,92},{35,74},{38,85},{40,56},
|
|
|
|
{42,87},{43,65},{45,59},{48,79},{49,93},{51,89},{55,95},{61,76},{67,83},
|
|
|
|
{77,80} };
|
|
|
|
|
|
|
|
// Map flavour to bytes per second
|
|
|
|
static int sipr_fl2bps[4] = {813, 1062, 625, 2000}; // 6.5, 8.5, 5, 16 kbit per second
|
2003-04-30 19:54:54 +00:00
|
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
unsigned short version;
|
|
|
|
unsigned int dotranum;
|
|
|
|
unsigned int data_size;
|
|
|
|
unsigned short version2;
|
|
|
|
unsigned int hdr_size;
|
|
|
|
unsigned short codec_flavor;
|
|
|
|
unsigned int coded_framesize;
|
|
|
|
unsigned short sub_packet_h;
|
|
|
|
unsigned short frame_size;
|
|
|
|
unsigned short sub_packet_size;
|
|
|
|
char genr[4];
|
2006-01-15 21:58:10 +00:00
|
|
|
unsigned char *audio_buf;
|
2003-04-30 19:54:54 +00:00
|
|
|
} ra_priv_t;
|
|
|
|
|
|
|
|
|
|
|
|
|
2005-08-05 19:57:47 +00:00
|
|
|
static int ra_check_file(demuxer_t* demuxer)
|
2003-04-30 19:54:54 +00:00
|
|
|
{
|
|
|
|
unsigned int chunk_id;
|
|
|
|
|
|
|
|
chunk_id = stream_read_dword_le(demuxer->stream);
|
|
|
|
if (chunk_id == FOURCC_DOTRA)
|
2005-08-05 19:57:47 +00:00
|
|
|
return DEMUXER_TYPE_REALAUDIO;
|
2003-04-30 19:54:54 +00:00
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// return value:
|
|
|
|
// 0 = EOF or no stream found
|
|
|
|
// 1 = successfully read a packet
|
2005-08-05 19:57:47 +00:00
|
|
|
static int demux_ra_fill_buffer(demuxer_t *demuxer, demux_stream_t *dsds)
|
2003-04-30 19:54:54 +00:00
|
|
|
{
|
|
|
|
ra_priv_t *ra_priv = demuxer->priv;
|
|
|
|
int len;
|
|
|
|
demux_stream_t *ds = demuxer->audio;
|
|
|
|
sh_audio_t *sh = ds->sh;
|
|
|
|
WAVEFORMATEX *wf = sh->wf;
|
|
|
|
demux_packet_t *dp;
|
2005-12-09 16:25:37 +00:00
|
|
|
int x, y;
|
2003-04-30 19:54:54 +00:00
|
|
|
|
|
|
|
if (demuxer->stream->eof)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
len = wf->nBlockAlign;
|
|
|
|
demuxer->filepos = stream_tell(demuxer->stream);
|
|
|
|
|
2006-01-15 21:58:10 +00:00
|
|
|
if ((sh->format == FOURCC_288) || (sh->format == FOURCC_SIPR)) {
|
|
|
|
if (sh->format == FOURCC_SIPR) {
|
|
|
|
int n;
|
|
|
|
int bs = ra_priv->sub_packet_h * ra_priv->frame_size * 2 / 96; // nibbles per subpacket
|
|
|
|
stream_read(demuxer->stream, ra_priv->audio_buf, ra_priv->sub_packet_h * ra_priv->frame_size);
|
|
|
|
// Perform reordering
|
|
|
|
for(n = 0; n < 38; n++) {
|
|
|
|
int j;
|
|
|
|
int i = bs * sipr_swaps[n][0];
|
|
|
|
int o = bs * sipr_swaps[n][1];
|
|
|
|
// swap nibbles of block 'i' with 'o' TODO: optimize
|
|
|
|
for(j = 0; j < bs; j++) {
|
|
|
|
int x = (i & 1) ? (ra_priv->audio_buf[i >> 1] >> 4) : (ra_priv->audio_buf[i >> 1] & 0x0F);
|
|
|
|
int y = (o & 1) ? (ra_priv->audio_buf[o >> 1] >> 4) : (ra_priv->audio_buf[o >> 1] & 0x0F);
|
|
|
|
if(o & 1)
|
|
|
|
ra_priv->audio_buf[o >> 1] = (ra_priv->audio_buf[o >> 1] & 0x0F) | (x << 4);
|
|
|
|
else
|
|
|
|
ra_priv->audio_buf[o >> 1] = (ra_priv->audio_buf[o >> 1] & 0xF0) | x;
|
|
|
|
if(i & 1)
|
|
|
|
ra_priv->audio_buf[i >> 1] = (ra_priv->audio_buf[i >> 1] & 0x0F) | (y << 4);
|
|
|
|
else
|
|
|
|
ra_priv->audio_buf[i >> 1] = (ra_priv->audio_buf[i >> 1] & 0xF0) | y;
|
|
|
|
++i; ++o;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2005-12-09 16:25:37 +00:00
|
|
|
for (y = 0; y < ra_priv->sub_packet_h; y++)
|
|
|
|
for (x = 0; x < ra_priv->sub_packet_h / 2; x++)
|
|
|
|
stream_read(demuxer->stream, ra_priv->audio_buf + x * 2 *ra_priv->frame_size +
|
|
|
|
y * ra_priv->coded_framesize, ra_priv->coded_framesize);
|
2006-01-15 21:58:10 +00:00
|
|
|
}
|
2005-12-09 16:25:37 +00:00
|
|
|
// Release all the audio packets
|
|
|
|
for (x = 0; x < ra_priv->sub_packet_h * ra_priv->frame_size / len; x++) {
|
|
|
|
dp = new_demux_packet(len);
|
|
|
|
memcpy(dp->buffer, ra_priv->audio_buf + x * len, len);
|
|
|
|
dp->pts = x ? 0 : demuxer->filepos / ra_priv->data_size;
|
|
|
|
dp->pos = demuxer->filepos; // all equal
|
|
|
|
dp->flags = x ? 0 : 0x10; // Mark first packet as keyframe
|
|
|
|
ds_add_packet(ds, dp);
|
|
|
|
}
|
|
|
|
} else {
|
2003-04-30 19:54:54 +00:00
|
|
|
dp = new_demux_packet(len);
|
|
|
|
stream_read(demuxer->stream, dp->buffer, len);
|
|
|
|
|
2003-04-30 20:17:35 +00:00
|
|
|
dp->pts = demuxer->filepos / ra_priv->data_size;
|
2003-04-30 19:54:54 +00:00
|
|
|
dp->pos = demuxer->filepos;
|
|
|
|
dp->flags = 0;
|
|
|
|
ds_add_packet(ds, dp);
|
2005-12-09 16:25:37 +00:00
|
|
|
}
|
2003-04-30 19:54:54 +00:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-03-27 17:25:41 +00:00
|
|
|
extern void print_wave_header(WAVEFORMATEX *h, int verbose_level);
|
2003-04-30 19:54:54 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2005-08-05 19:57:47 +00:00
|
|
|
static demuxer_t* demux_open_ra(demuxer_t* demuxer)
|
2003-04-30 19:54:54 +00:00
|
|
|
{
|
|
|
|
ra_priv_t* ra_priv = demuxer->priv;
|
|
|
|
sh_audio_t *sh;
|
|
|
|
int i;
|
|
|
|
char *buf;
|
|
|
|
|
2006-07-13 16:41:13 +00:00
|
|
|
if ((ra_priv = malloc(sizeof(ra_priv_t))) == NULL) {
|
2003-04-30 19:54:54 +00:00
|
|
|
mp_msg(MSGT_DEMUX, MSGL_ERR, "[RealAudio] Can't allocate memory for private data.\n");
|
2003-06-01 20:27:32 +00:00
|
|
|
return 0;
|
2003-04-30 19:54:54 +00:00
|
|
|
}
|
|
|
|
memset(ra_priv, 0, sizeof(ra_priv_t));
|
|
|
|
|
|
|
|
demuxer->priv = ra_priv;
|
|
|
|
sh = new_sh_audio(demuxer, 0);
|
|
|
|
demuxer->audio->id = 0;
|
|
|
|
sh->ds=demuxer->audio;
|
|
|
|
demuxer->audio->sh = sh;
|
|
|
|
|
|
|
|
ra_priv->version = stream_read_word(demuxer->stream);
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] File version: %d\n", ra_priv->version);
|
|
|
|
if ((ra_priv->version < 3) || (ra_priv->version > 4)) {
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] ra version %d is not supported yet, please "
|
|
|
|
"contact MPlayer developers\n", ra_priv->version);
|
2003-06-01 20:27:32 +00:00
|
|
|
return 0;
|
2003-04-30 19:54:54 +00:00
|
|
|
}
|
|
|
|
if (ra_priv->version == 3) {
|
|
|
|
ra_priv->hdr_size = stream_read_word(demuxer->stream);
|
|
|
|
stream_skip(demuxer->stream, 10);
|
|
|
|
ra_priv->data_size = stream_read_dword(demuxer->stream);
|
|
|
|
} else {
|
|
|
|
stream_skip(demuxer->stream, 2);
|
|
|
|
ra_priv->dotranum = stream_read_dword(demuxer->stream);
|
|
|
|
ra_priv->data_size = stream_read_dword(demuxer->stream);
|
|
|
|
ra_priv->version2 = stream_read_word(demuxer->stream);
|
|
|
|
ra_priv->hdr_size = stream_read_dword(demuxer->stream);
|
|
|
|
ra_priv->codec_flavor = stream_read_word(demuxer->stream);
|
2006-01-15 21:58:10 +00:00
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] Flavor: %d\n", ra_priv->codec_flavor);
|
2003-04-30 19:54:54 +00:00
|
|
|
ra_priv->coded_framesize = stream_read_dword(demuxer->stream);
|
2006-01-15 21:58:10 +00:00
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] Coded frame size: %d\n", ra_priv->coded_framesize);
|
2003-04-30 19:54:54 +00:00
|
|
|
stream_skip(demuxer->stream, 4); // data size?
|
|
|
|
stream_skip(demuxer->stream, 8);
|
|
|
|
ra_priv->sub_packet_h = stream_read_word(demuxer->stream);
|
2006-01-15 21:58:10 +00:00
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] Sub packet h: %d\n", ra_priv->sub_packet_h);
|
2003-04-30 19:54:54 +00:00
|
|
|
ra_priv->frame_size = stream_read_word(demuxer->stream);
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] Frame size: %d\n", ra_priv->frame_size);
|
|
|
|
ra_priv->sub_packet_size = stream_read_word(demuxer->stream);
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] Sub packet size: %d\n", ra_priv->sub_packet_size);
|
|
|
|
stream_skip(demuxer->stream, 2);
|
|
|
|
sh->samplerate = stream_read_word(demuxer->stream);
|
|
|
|
stream_skip(demuxer->stream, 2);
|
|
|
|
sh->samplesize = stream_read_word(demuxer->stream);
|
|
|
|
sh->channels = stream_read_word(demuxer->stream);
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] %d channel, %d bit, %dHz\n", sh->channels,
|
|
|
|
sh->samplesize, sh->samplerate);
|
|
|
|
i = stream_read_char(demuxer->stream);
|
2005-08-30 23:59:21 +00:00
|
|
|
ra_priv->genr[0] = stream_read_char(demuxer->stream);
|
|
|
|
ra_priv->genr[1] = stream_read_char(demuxer->stream);
|
|
|
|
ra_priv->genr[2] = stream_read_char(demuxer->stream);
|
|
|
|
ra_priv->genr[3] = stream_read_char(demuxer->stream);
|
2003-04-30 19:54:54 +00:00
|
|
|
if (i != 4) {
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] Genr size is not 4 (%d), please report to "
|
|
|
|
"MPlayer developers\n", i);
|
|
|
|
stream_skip(demuxer->stream, i - 4);
|
|
|
|
}
|
|
|
|
i = stream_read_char(demuxer->stream);
|
|
|
|
sh->format = stream_read_dword_le(demuxer->stream);
|
|
|
|
if (i != 4) {
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] FourCC size is not 4 (%d), please report to "
|
|
|
|
"MPlayer developers\n", i);
|
|
|
|
stream_skip(demuxer->stream, i - 4);
|
|
|
|
}
|
|
|
|
stream_skip(demuxer->stream, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((i = stream_read_char(demuxer->stream)) != 0) {
|
|
|
|
buf = malloc(i+1);
|
|
|
|
stream_read(demuxer->stream, buf, i);
|
|
|
|
buf[i] = 0;
|
|
|
|
demux_info_add(demuxer, "Title", buf);
|
|
|
|
free(buf);
|
|
|
|
}
|
|
|
|
if ((i = stream_read_char(demuxer->stream)) != 0) {
|
|
|
|
buf = malloc(i+1);
|
|
|
|
stream_read(demuxer->stream, buf, i);
|
|
|
|
buf[i] = 0;
|
|
|
|
demux_info_add(demuxer, "Author", buf);
|
|
|
|
free(buf);
|
|
|
|
}
|
|
|
|
if ((i = stream_read_char(demuxer->stream)) != 0) {
|
|
|
|
buf = malloc(i+1);
|
|
|
|
stream_read(demuxer->stream, buf, i);
|
|
|
|
buf[i] = 0;
|
|
|
|
demux_info_add(demuxer, "Copyright", buf);
|
|
|
|
free(buf);
|
|
|
|
}
|
|
|
|
|
2005-03-03 21:13:30 +00:00
|
|
|
if ((i = stream_read_char(demuxer->stream)) != 0) {
|
|
|
|
buf = malloc(i+1);
|
|
|
|
stream_read(demuxer->stream, buf, i);
|
|
|
|
buf[i] = 0;
|
|
|
|
demux_info_add(demuxer, "Comment", buf);
|
|
|
|
free(buf);
|
|
|
|
}
|
2003-04-30 19:54:54 +00:00
|
|
|
|
|
|
|
if (ra_priv->version == 3) {
|
2005-03-03 23:02:45 +00:00
|
|
|
if(ra_priv->hdr_size + 8 > stream_tell(demuxer->stream)) {
|
2003-04-30 19:54:54 +00:00
|
|
|
stream_skip(demuxer->stream, 1);
|
|
|
|
i = stream_read_char(demuxer->stream);
|
|
|
|
sh->format = stream_read_dword_le(demuxer->stream);
|
|
|
|
if (i != 4) {
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] FourCC size is not 4 (%d), please report to "
|
|
|
|
"MPlayer developers\n", i);
|
|
|
|
stream_skip(demuxer->stream, i - 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sh->format != FOURCC_LPCJ) {
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] Version 3 with FourCC %8x, please report to "
|
|
|
|
"MPlayer developers\n", sh->format);
|
|
|
|
}
|
2005-03-03 23:02:45 +00:00
|
|
|
} else
|
|
|
|
// If a stream does not have fourcc, let's assume it's 14.4
|
|
|
|
sh->format = FOURCC_LPCJ;
|
2003-04-30 19:54:54 +00:00
|
|
|
|
|
|
|
sh->channels = 1;
|
|
|
|
sh->samplesize = 16;
|
|
|
|
sh->samplerate = 8000;
|
|
|
|
ra_priv->frame_size = 240;
|
|
|
|
sh->format = FOURCC_144;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Fill WAVEFORMATEX */
|
|
|
|
sh->wf = malloc(sizeof(WAVEFORMATEX));
|
|
|
|
memset(sh->wf, 0, sizeof(WAVEFORMATEX));
|
|
|
|
sh->wf->nChannels = sh->channels;
|
|
|
|
sh->wf->wBitsPerSample = sh->samplesize;
|
|
|
|
sh->wf->nSamplesPerSec = sh->samplerate;
|
|
|
|
sh->wf->nAvgBytesPerSec = sh->samplerate*sh->samplesize/8;
|
|
|
|
sh->wf->nBlockAlign = ra_priv->frame_size;
|
|
|
|
sh->wf->cbSize = 0;
|
|
|
|
sh->wf->wFormatTag = sh->format;
|
|
|
|
|
|
|
|
switch (sh->format) {
|
|
|
|
case FOURCC_144:
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"Audio: 14_4\n");
|
2005-12-09 16:25:37 +00:00
|
|
|
sh->wf->nBlockAlign = 0x14;
|
2003-04-30 19:54:54 +00:00
|
|
|
break;
|
|
|
|
case FOURCC_288:
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"Audio: 28_8\n");
|
2005-12-09 16:25:37 +00:00
|
|
|
sh->wf->nBlockAlign = ra_priv->coded_framesize;
|
2006-05-14 15:51:05 +00:00
|
|
|
ra_priv->audio_buf = calloc(ra_priv->sub_packet_h, ra_priv->frame_size);
|
2003-04-30 19:54:54 +00:00
|
|
|
break;
|
|
|
|
case FOURCC_DNET:
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"Audio: DNET -> AC3\n");
|
|
|
|
break;
|
2006-01-15 21:58:10 +00:00
|
|
|
case FOURCC_SIPR:
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"Audio: SIPR\n");
|
|
|
|
sh->wf->nBlockAlign = ra_priv->coded_framesize;
|
|
|
|
sh->wf->nAvgBytesPerSec = sipr_fl2bps[ra_priv->codec_flavor];
|
2006-05-14 15:51:05 +00:00
|
|
|
ra_priv->audio_buf = calloc(ra_priv->sub_packet_h, ra_priv->frame_size);
|
2006-01-15 21:58:10 +00:00
|
|
|
break;
|
2003-04-30 19:54:54 +00:00
|
|
|
default:
|
|
|
|
mp_msg(MSGT_DEMUX,MSGL_V,"Audio: Unknown (%d)\n", sh->format);
|
|
|
|
}
|
|
|
|
|
2006-03-27 17:25:41 +00:00
|
|
|
print_wave_header(sh->wf, MSGL_V);
|
2003-04-30 19:54:54 +00:00
|
|
|
|
|
|
|
/* disable seeking */
|
|
|
|
demuxer->seekable = 0;
|
|
|
|
|
|
|
|
if(!ds_fill_buffer(demuxer->audio))
|
|
|
|
mp_msg(MSGT_DEMUXER,MSGL_INFO,"[RealAudio] No data.\n");
|
2003-06-01 20:27:32 +00:00
|
|
|
|
2005-08-05 19:57:47 +00:00
|
|
|
return demuxer;
|
2003-04-30 19:54:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2005-08-05 19:57:47 +00:00
|
|
|
static void demux_close_ra(demuxer_t *demuxer)
|
2003-04-30 19:54:54 +00:00
|
|
|
{
|
|
|
|
ra_priv_t* ra_priv = demuxer->priv;
|
|
|
|
|
2005-12-09 16:25:37 +00:00
|
|
|
if (ra_priv) {
|
|
|
|
if (ra_priv->audio_buf)
|
|
|
|
free (ra_priv->audio_buf);
|
2003-04-30 19:54:54 +00:00
|
|
|
free(ra_priv);
|
2005-12-09 16:25:37 +00:00
|
|
|
}
|
2003-04-30 19:54:54 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* please upload RV10 samples WITH INDEX CHUNK */
|
2006-02-17 01:57:41 +00:00
|
|
|
int demux_seek_ra(demuxer_t *demuxer, float rel_seek_secs, float audio_delay, int flags)
|
2003-04-30 19:54:54 +00:00
|
|
|
{
|
|
|
|
real_priv_t *priv = demuxer->priv;
|
|
|
|
demux_stream_t *d_audio = demuxer->audio;
|
|
|
|
sh_audio_t *sh_audio = d_audio->sh;
|
|
|
|
int aid = d_audio->id;
|
|
|
|
int next_offset = 0;
|
|
|
|
int rel_seek_frames = 0;
|
|
|
|
int streams = 0;
|
|
|
|
|
|
|
|
return stream_seek(demuxer->stream, next_offset);
|
|
|
|
}
|
|
|
|
#endif
|
2005-08-05 19:57:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
demuxer_desc_t demuxer_desc_realaudio = {
|
|
|
|
"Realaudio demuxer",
|
|
|
|
"realaudio",
|
|
|
|
"REALAUDIO",
|
|
|
|
"Roberto Togni",
|
|
|
|
"handles old audio only .ra files",
|
|
|
|
DEMUXER_TYPE_REALAUDIO,
|
|
|
|
1, // safe autodetect
|
|
|
|
ra_check_file,
|
|
|
|
demux_ra_fill_buffer,
|
|
|
|
demux_open_ra,
|
|
|
|
demux_close_ra,
|
|
|
|
NULL,
|
|
|
|
NULL
|
|
|
|
};
|