Skip to content

Commit

Permalink
Switch to new channel layout API. (#59)
Browse files Browse the repository at this point in the history
  • Loading branch information
toots authored Jul 9, 2024
1 parent 1911d25 commit 5406339
Show file tree
Hide file tree
Showing 36 changed files with 544 additions and 282 deletions.
5 changes: 4 additions & 1 deletion .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -12,10 +12,13 @@ jobs:
strategy:
fail-fast: false
matrix:
os: [macos-latest, ubuntu-latest]
os: [macos-latest, ubuntu-24.04]
compiler: [4.14.2, 5.2.0]
steps:
- name: Build and test module
uses: savonet/build-and-test-ocaml-module@main
with:
ocaml-compiler: ${{ matrix.compiler }}
- name: Run CI tests
run: |
opam exec dune build @citest
1 change: 1 addition & 0 deletions CHANGES
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
* Added support to get all streams from output container
* Removed unreliable `Av.was_keyframe`
* Added `on_keyframe` to `Av.write_frame`.
* Switched to new channel layout API.

1.1.11 (2024-03-19)
======
Expand Down
20 changes: 5 additions & 15 deletions av/av.ml
Original file line number Diff line number Diff line change
Expand Up @@ -318,30 +318,20 @@ external new_audio_stream :
_ container ->
int ->
[ `Encoder ] Avcodec.Audio.t ->
int ->
Channel_layout.t ->
(string * string) array ->
int * string array = "ocaml_av_new_audio_stream"

let new_audio_stream ?opts ?channels ?channel_layout ~sample_rate ~sample_format
let new_audio_stream ?opts ~channel_layout ~sample_rate ~sample_format
~time_base ~codec container =
let opts =
mk_audio_opts ?opts ?channels ?channel_layout ~sample_rate ~sample_format
~time_base ()
in
let channels =
match (channels, channel_layout) with
| Some n, _ -> n
| None, Some layout -> Avutil.Channel_layout.get_nb_channels layout
| None, None ->
raise
(Error
(`Failure
"At least one of channels or channel_layout must be passed!"))
mk_audio_opts ?opts ~channel_layout ~sample_rate ~sample_format ~time_base
()
in
let ret, unused =
new_audio_stream container
(Sample_format.get_id sample_format)
codec channels (mk_opts_array opts)
codec channel_layout (mk_opts_array opts)
in
filter_opts unused opts;
mk_stream container ret
Expand Down
5 changes: 1 addition & 4 deletions av/av.mli
Original file line number Diff line number Diff line change
Expand Up @@ -268,8 +268,6 @@ val initialize_stream_copy :
After returning, if [opts] was passed, unused options are left in the hash
table.
At least one of [channels] or [channel_layout] must be passed.
Frames passed to this stream for encoding must have a PTS set according to
the given [time_base]. [1/sample_rate] is usually a good value for the
[time_base].
Expand All @@ -285,8 +283,7 @@ val initialize_stream_copy :
Raise Error if the opening failed. *)
val new_audio_stream :
?opts:opts ->
?channels:int ->
?channel_layout:Channel_layout.t ->
channel_layout:Channel_layout.t ->
sample_rate:int ->
sample_format:Avutil.Sample_format.t ->
time_base:Avutil.rational ->
Expand Down
33 changes: 23 additions & 10 deletions av/av_stubs.c
Original file line number Diff line number Diff line change
Expand Up @@ -357,8 +357,13 @@ static int ocaml_avio_read_callback(void *private, uint8_t *buf, int buf_size) {
return Int_val(res);
}

#if LIBAVFORMAT_VERSION_MAJOR < 61
static int ocaml_avio_write_callback(void *private, uint8_t *buf,
int buf_size) {
#else
static int ocaml_avio_write_callback(void *private, const uint8_t *buf,
int buf_size) {
#endif
value buffer, res;
avio_t *avio = (avio_t *)private;
int ret;
Expand Down Expand Up @@ -481,7 +486,11 @@ CAMLprim value ocaml_av_create_io(value bufsize, value _read_cb,
CAMLlocal1(ret);

int (*read_cb)(void *opaque, uint8_t *buf, int buf_size) = NULL;
#if LIBAVFORMAT_VERSION_MAJOR < 61
int (*write_cb)(void *opaque, uint8_t *buf, int buf_size) = NULL;
#else
int (*write_cb)(void *opaque, const uint8_t *buf, int buf_size) = NULL;
#endif
int64_t (*seek_cb)(void *opaque, int64_t offset, int whence) = NULL;
int write_flag = 0;
unsigned char *buffer;
Expand Down Expand Up @@ -1717,7 +1726,7 @@ static stream_t *new_stream(av_t *av, const AVCodec *codec) {

AVStream *avstream = avformat_new_stream(av->format_context, codec);
if (!avstream) {
free(stream);
free_stream(stream);
caml_raise_out_of_memory();
}

Expand Down Expand Up @@ -1773,18 +1782,21 @@ static void init_stream_encoder(AVBufferRef *device_ctx, AVBufferRef *frame_ctx,
}

static stream_t *new_audio_stream(av_t *av, enum AVSampleFormat sample_fmt,
int channels, const AVCodec *codec,
AVChannelLayout *channel_layout,
const AVCodec *codec,
AVDictionary **options) {
stream_t *stream = new_stream(av, codec);
int ret;

AVCodecContext *enc_ctx = stream->codec_context;

enc_ctx->sample_fmt = sample_fmt;
enc_ctx->channels = channels;
// Detect new API
#ifdef AV_CHANNEL_LAYOUT_MONO
av_channel_layout_default(&enc_ctx->ch_layout, channels);
#endif
ret = av_channel_layout_copy(&enc_ctx->ch_layout, channel_layout);

if (ret < 0) {
free_stream(stream);
ocaml_avutil_raise_error(ret);
}

init_stream_encoder(NULL, NULL, av, stream, options);

Expand Down Expand Up @@ -1818,7 +1830,7 @@ CAMLprim value ocaml_av_initialize_stream_copy(value _av, value _stream_index,
}

CAMLprim value ocaml_av_new_audio_stream(value _av, value _sample_fmt,
value _codec, value _channels,
value _codec, value _channel_layout,
value _opts) {
CAMLparam2(_av, _opts);
CAMLlocal2(ans, unused);
Expand All @@ -1840,8 +1852,9 @@ CAMLprim value ocaml_av_new_audio_stream(value _av, value _sample_fmt,
}
}

stream_t *stream = new_audio_stream(Av_val(_av), Int_val(_sample_fmt),
Int_val(_channels), codec, &options);
stream_t *stream =
new_audio_stream(Av_val(_av), Int_val(_sample_fmt),
AVChannelLayout_val(_channel_layout), codec, &options);

// Return unused keys
count = av_dict_count(options);
Expand Down
19 changes: 12 additions & 7 deletions av/av_stubs.h
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
#ifndef _AV_STUBS_H_
#ifndef _AV_STUBS_H_
#define _AV_STUBS_H_

#include <caml/mlvalues.h>
Expand All @@ -15,19 +15,24 @@ AVFormatContext *ocaml_av_get_format_context(value *p_av);

/***** AVInputFormat *****/

#define InputFormat_val(v) (*(avioformat_const AVInputFormat**)Data_abstract_val(v))
#define InputFormat_val(v) \
(*(avioformat_const AVInputFormat **)Data_abstract_val(v))

void value_of_inputFormat(avioformat_const AVInputFormat *inputFormat, value * p_value);
void value_of_inputFormat(avioformat_const AVInputFormat *inputFormat,
value *p_value);

/***** AVOutputFormat *****/

#define OutputFormat_val(v) (*(avioformat_const AVOutputFormat**)Data_abstract_val(v))
#define OutputFormat_val(v) \
(*(avioformat_const AVOutputFormat **)Data_abstract_val(v))

value value_of_outputFormat(avioformat_const AVOutputFormat *outputFormat);

/***** Control message *****/
value * ocaml_av_get_control_message_callback(struct AVFormatContext *ctx);
value *ocaml_av_get_control_message_callback(struct AVFormatContext *ctx);

void ocaml_av_set_control_message_callback(value *p_av, av_format_control_message c_callback, value *p_ocaml_callback);
void ocaml_av_set_control_message_callback(value *p_av,
av_format_control_message c_callback,
value *p_ocaml_callback);

#endif // _AV_STUBS_H_
#endif // _AV_STUBS_H_
2 changes: 1 addition & 1 deletion av/config/discover.ml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
module C = Configurator.V1

let packages = [("avutil", "55.78.100"); ("avformat", "57.83.100")]
let packages = [("avutil", "57.24.100"); ("avformat", "57.83.100")]

let () =
C.main ~name:"ffmpeg-av-pkg-config" (fun c ->
Expand Down
26 changes: 10 additions & 16 deletions avcodec/avcodec.ml
Original file line number Diff line number Diff line change
Expand Up @@ -329,7 +329,11 @@ module Audio = struct
let find_best_channel_layout codec default =
try
let channel_layouts = get_supported_channel_layouts codec in
if List.mem default channel_layouts then default
if
List.exists
(fun layout -> Avutil.Channel_layout.compare layout default)
channel_layouts
then default
else (match channel_layouts with h :: _ -> h | [] -> default)
with Not_found -> default

Expand Down Expand Up @@ -384,31 +388,21 @@ module Audio = struct
external create_encoder :
int ->
[ `Encoder ] t ->
int ->
Channel_layout.t ->
(string * string) array ->
audio encoder * string array = "ocaml_avcodec_create_audio_encoder"

let create_encoder ?opts ?channels ?channel_layout ~sample_rate ~sample_format
let create_encoder ?opts ~channel_layout ~sample_rate ~sample_format
~time_base codec =
let opts = opts_default opts in
let _opts =
mk_audio_opts ~opts ?channels ?channel_layout ~sample_rate ~sample_format
~time_base ()
in
let channels =
match (channels, channel_layout) with
| Some n, _ -> n
| None, Some layout -> Avutil.Channel_layout.get_nb_channels layout
| None, None ->
raise
(Error
(`Failure
"At least one of channels or channel_layout must be passed!"))
mk_audio_opts ~opts ~channel_layout ~sample_rate ~sample_format ~time_base
()
in
let encoder, unused =
create_encoder
(Sample_format.get_id sample_format)
codec channels (mk_opts_array _opts)
codec channel_layout (mk_opts_array _opts)
in
filter_opts unused opts;
encoder
Expand Down
3 changes: 1 addition & 2 deletions avcodec/avcodec.mli
Original file line number Diff line number Diff line change
Expand Up @@ -238,8 +238,7 @@ module Audio : sig
Raise Error if the encoder creation failed. *)
val create_encoder :
?opts:opts ->
?channels:int ->
?channel_layout:Channel_layout.t ->
channel_layout:Channel_layout.t ->
sample_rate:int ->
sample_format:Avutil.Sample_format.t ->
time_base:Avutil.rational ->
Expand Down
50 changes: 13 additions & 37 deletions avcodec/avcodec_stubs.c
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,8 @@
#include "hw_config_method_stubs.h"
#include "media_types_stubs.h"

#include <libavutil/replaygain.h>
#include <libavcodec/bsf.h>
#include <libavutil/replaygain.h>

#ifndef AV_PKT_FLAG_DISPOSABLE
#define AV_PKT_FLAG_DISPOSABLE 0x0010
Expand Down Expand Up @@ -662,7 +662,8 @@ CAMLprim value ocaml_avcodec_encoder_time_base(value _encoder) {
}

CAMLprim value ocaml_avcodec_create_audio_encoder(value _sample_fmt,
value _codec, value _channels,
value _codec,
value _channel_layout,
value _opts) {
CAMLparam2(_opts, _codec);
CAMLlocal3(ret, ans, unused);
Expand Down Expand Up @@ -700,11 +701,11 @@ CAMLprim value ocaml_avcodec_create_audio_encoder(value _sample_fmt,
}

ctx->codec_context->sample_fmt = Int_val(_sample_fmt);
ctx->codec_context->channels = Int_val(_channels);
// Detect new API
#ifdef AV_CHANNEL_LAYOUT_MONO
av_channel_layout_default(&ctx->codec_context->ch_layout, Int_val(_channels));
#endif

err = av_channel_layout_copy(&ctx->codec_context->ch_layout,
AVChannelLayout_val(_channel_layout));
if (err < 0)
ocaml_avutil_raise_error(err);

// Open the codec
caml_release_runtime_system();
Expand Down Expand Up @@ -1290,9 +1291,9 @@ CAMLprim value ocaml_avcodec_get_supported_channel_layouts(value _codec) {
List_init(list);
const AVCodec *codec = AvCodec_val(_codec);

if (codec->channel_layouts) {
for (i = 0; codec->channel_layouts[i] != 0; i++)
List_add(list, cons, Val_ChannelLayout(codec->channel_layouts[i]));
if (codec->ch_layouts) {
for (i = 0; codec->ch_layouts[i].nb_channels != 0; i++)
List_add(list, cons, value_of_channel_layout(&codec->ch_layouts[i]));
}

CAMLreturn(list);
Expand Down Expand Up @@ -1338,16 +1339,12 @@ CAMLprim value ocaml_avcodec_parameters_get_channel_layout(value _cp) {
CAMLparam1(_cp);
AVCodecParameters *cp = CodecParameters_val(_cp);

if (cp->channel_layout == 0) {
cp->channel_layout = av_get_default_channel_layout(cp->channels);
}

CAMLreturn(Val_ChannelLayout(cp->channel_layout));
CAMLreturn(value_of_channel_layout(&cp->ch_layout));
}

CAMLprim value ocaml_avcodec_parameters_get_nb_channels(value _cp) {
CAMLparam1(_cp);
CAMLreturn(Val_int(CodecParameters_val(_cp)->channels));
CAMLreturn(Val_int(CodecParameters_val(_cp)->ch_layout.nb_channels));
}

CAMLprim value ocaml_avcodec_parameters_get_sample_format(value _cp) {
Expand All @@ -1361,27 +1358,6 @@ CAMLprim value ocaml_avcodec_parameters_get_sample_rate(value _cp) {
CAMLreturn(Val_int(CodecParameters_val(_cp)->sample_rate));
}

CAMLprim value ocaml_avcodec_parameters_audio_copy(value _codec_id,
value _channel_layout,
value _sample_format,
value _sample_rate,
value _cp) {
CAMLparam4(_codec_id, _channel_layout, _sample_format, _cp);
CAMLlocal1(ans);

value_of_codec_parameters_copy(CodecParameters_val(_cp), &ans);

AVCodecParameters *dst = CodecParameters_val(ans);

dst->codec_id = AudioCodecID_val(_codec_id);
dst->channel_layout = ChannelLayout_val(_channel_layout);
dst->channels = av_get_channel_layout_nb_channels(dst->channel_layout);
dst->format = SampleFormat_val(_sample_format);
dst->sample_rate = Int_val(_sample_rate);

CAMLreturn(ans);
}

/**** Video codec ID ****/

CAMLprim value ocaml_avcodec_get_video_codec_id_name(value _codec_id) {
Expand Down
2 changes: 1 addition & 1 deletion avcodec/config/discover.ml
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ let () =
| Some pc -> (
match
C.Pkg_config.query_expr_err pc ~package:"libavcodec"
~expr:"libavcodec >= 58.87.100"
~expr:"libavcodec >= 59.24.100"
with
| Error msg -> failwith msg
| Ok deps -> deps)
Expand Down
Loading

0 comments on commit 5406339

Please sign in to comment.