2 * Wideband Single-bit Data (WSD) demuxer
3 * Copyright (c) 2014 Peter Ross
5 * This file is part of FFmpeg.
7 * FFmpeg is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * FFmpeg is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with FFmpeg; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22 #include "libavutil/channel_layout.h"
23 #include "libavutil/intreadwrite.h"
24 #include "libavutil/mem.h"
25 #include "libavutil/timecode.h"
27 #include "avio_internal.h"
31 static int wsd_probe(const AVProbeData
*p
)
33 if (p
->buf_size
< 45 || memcmp(p
->buf
, "1bit", 4) ||
34 !AV_RB32(p
->buf
+ 36) || !p
->buf
[44] ||
35 (p
->buf
[0] >= 0x10 && (AV_RB32(p
->buf
+ 20) < 0x80 || AV_RB32(p
->buf
+ 24) < 0x80)))
37 return AVPROBE_SCORE_MAX
;
40 static int empty_string(const char *buf
, unsigned size
)
49 static int wsd_to_av_channel_layoyt(AVFormatContext
*s
, int bit
)
52 case 2: return AV_CH_BACK_RIGHT
;
54 avpriv_request_sample(s
, "Rr-middle");
56 case 4: return AV_CH_BACK_CENTER
;
58 avpriv_request_sample(s
, "Lr-middle");
60 case 6: return AV_CH_BACK_LEFT
;
61 case 24: return AV_CH_LOW_FREQUENCY
;
62 case 26: return AV_CH_FRONT_RIGHT
;
63 case 27: return AV_CH_FRONT_RIGHT_OF_CENTER
;
64 case 28: return AV_CH_FRONT_CENTER
;
65 case 29: return AV_CH_FRONT_LEFT_OF_CENTER
;
66 case 30: return AV_CH_FRONT_LEFT
;
68 av_log(s
, AV_LOG_WARNING
, "reserved channel assignment\n");
74 static int get_metadata(AVFormatContext
*s
, const char *const tag
, const unsigned size
)
79 return AVERROR(ENOMEM
);
81 buf
= av_malloc(size
+ 1);
83 return AVERROR(ENOMEM
);
85 if ((ret
= avio_read(s
->pb
, buf
, size
)) < 0) {
90 if (empty_string(buf
, size
)) {
96 av_dict_set(&s
->metadata
, tag
, buf
, AV_DICT_DONT_STRDUP_VAL
);
100 static int wsd_read_header(AVFormatContext
*s
)
102 AVIOContext
*pb
= s
->pb
;
105 uint32_t text_offset
, data_offset
, channel_assign
;
106 char playback_time
[AV_TIMECODE_STR_SIZE
];
108 st
= avformat_new_stream(s
, NULL
);
110 return AVERROR(ENOMEM
);
113 version
= avio_r8(pb
);
114 av_log(s
, AV_LOG_DEBUG
, "version: %i.%i\n", version
>> 4, version
& 0xF);
117 if (version
< 0x10) {
122 text_offset
= avio_rb32(pb
);
123 data_offset
= avio_rb32(pb
);
127 av_timecode_make_smpte_tc_string2(playback_time
, (AVRational
){1,1}, avio_rb32(pb
) & 0x00ffffffU
, 1, 1);
128 av_dict_set(&s
->metadata
, "playback_time", playback_time
, 0);
130 st
->codecpar
->codec_type
= AVMEDIA_TYPE_AUDIO
;
131 st
->codecpar
->codec_id
= AV_CODEC_ID_DSD_MSBF
;
132 st
->codecpar
->sample_rate
= avio_rb32(pb
) / 8;
134 st
->codecpar
->ch_layout
.nb_channels
= avio_r8(pb
) & 0xF;
135 st
->codecpar
->bit_rate
= (int64_t)st
->codecpar
->ch_layout
.nb_channels
*
136 st
->codecpar
->sample_rate
* 8LL;
137 if (!st
->codecpar
->ch_layout
.nb_channels
)
138 return AVERROR_INVALIDDATA
;
141 channel_assign
= avio_rb32(pb
);
142 if (!(channel_assign
& 1)) {
143 uint64_t ch_mask
= 0;
145 for (i
= 1; i
< 32; i
++)
146 if ((channel_assign
>> i
) & 1)
147 ch_mask
|= wsd_to_av_channel_layoyt(s
, i
);
148 av_channel_layout_from_mask(&st
->codecpar
->ch_layout
, ch_mask
);
153 avpriv_request_sample(s
, "emphasis");
155 if (avio_seek(pb
, text_offset
, SEEK_SET
) >= 0) {
156 get_metadata(s
, "title", 128);
157 get_metadata(s
, "composer", 128);
158 get_metadata(s
, "song_writer", 128);
159 get_metadata(s
, "artist", 128);
160 get_metadata(s
, "album", 128);
161 get_metadata(s
, "genre", 32);
162 get_metadata(s
, "date", 32);
163 get_metadata(s
, "location", 32);
164 get_metadata(s
, "comment", 512);
165 get_metadata(s
, "user", 512);
168 return avio_seek(pb
, data_offset
, SEEK_SET
);
171 const FFInputFormat ff_wsd_demuxer
= {
173 .p
.long_name
= NULL_IF_CONFIG_SMALL("Wideband Single-bit Data (WSD)"),
174 .p
.extensions
= "wsd",
175 .p
.flags
= AVFMT_GENERIC_INDEX
| AVFMT_NO_BYTE_SEEK
,
176 .p
.priv_class
= &ff_raw_demuxer_class
,
177 .read_probe
= wsd_probe
,
178 .read_header
= wsd_read_header
,
179 .read_packet
= ff_raw_read_partial_packet
,
180 .raw_codec_id
= AV_CODEC_ID_DSD_MSBF
,
181 .priv_data_size
= sizeof(FFRawDemuxerContext
),