Skip to content

Commit

Permalink
Add packged bigarray format for Swscale.
Browse files Browse the repository at this point in the history
  • Loading branch information
toots committed Oct 21, 2024
1 parent f204643 commit da98226
Show file tree
Hide file tree
Showing 3 changed files with 76 additions and 4 deletions.
9 changes: 8 additions & 1 deletion swscale/swscale.ml
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ type planes = (data * int) array
external scale : t -> planes -> int -> int -> planes -> int -> unit
= "ocaml_swscale_scale_byte" "ocaml_swscale_scale"

type vector_kind = Ba | Frm | Str
type vector_kind = PackedBa | Ba | Frm | Str

module type VideoData = sig
type t
Expand All @@ -34,6 +34,13 @@ module BigArray = struct
let vk = Ba
end

module PackedBigArray = struct
type plane = { plane_size : int; stride : int }
type t = { data : data; planes : plane array }

let vk = PackedBa
end

module Frame = struct
type t = video frame

Expand Down
12 changes: 10 additions & 2 deletions swscale/swscale.mli
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ val scale : t -> planes -> int -> int -> planes -> int -> unit

(**/**)

type vector_kind = Ba | Frm | Str
type vector_kind = PackedBa | Ba | Frm | Str

(**/**)

Expand Down Expand Up @@ -68,13 +68,21 @@ module Make (I : VideoData) (O : VideoData) : sig
val convert : t -> I.t -> O.t
end

(** Unsigned 8 bit bigarray. *)
(** Unsigned 8 bit bigarray split by planes. *)
module BigArray : sig
type t = planes

val vk : vector_kind
end

(** Unsigned 8 bit bigarray in a single packed array.. *)
module PackedBigArray : sig
type plane = { plane_size : int; stride : int }
type t = { data : data; planes : plane array }

val vk : vector_kind
end

(** Video frame. *)
module Frame : sig
type t = video frame
Expand Down
59 changes: 58 additions & 1 deletion swscale/swscale_stubs.c
Original file line number Diff line number Diff line change
Expand Up @@ -158,7 +158,7 @@ CAMLprim value ocaml_swscale_scale_byte(value *argv, int argn) {

/***** Contexts *****/

typedef enum _vector_kind { Ba, Frm, Str } vector_kind;
typedef enum _vector_kind { PackedBa, Ba, Frm, Str } vector_kind;

struct video_t {
int width;
Expand Down Expand Up @@ -235,6 +235,27 @@ static int get_in_pixels_ba(sws_t *sws, value *in_vector) {
CAMLreturnT(int, nb_planes);
}

static int get_in_pixels_packed_ba(sws_t *sws, value *in_vector) {
CAMLparam0();
CAMLlocal1(v);
uint8_t *data = Caml_ba_data_val(Field(*in_vector, 0));
int i, nb_planes = Wosize_val(Field(*in_vector, 1));
int stride, plane_size, offset = 0;

for (i = 0; i < nb_planes && i < 4; i++) {
v = Field(Field(*in_vector, 1), i);
plane_size = Int_val(Field(v, 0));
stride = Int_val(Field(v, 1));

sws->in.slice[i] = data + offset;
sws->in.stride[i] = stride;

offset += plane_size;
}

CAMLreturnT(int, nb_planes);
}

static int alloc_out_frame(sws_t *sws, value *out_vect, value *tmp) {
int ret;
AVFrame *frame = av_frame_alloc();
Expand Down Expand Up @@ -325,6 +346,38 @@ static int alloc_out_ba(sws_t *sws, value *out_vect, value *tmp) {
return 0;
}

static int alloc_out_packed_ba(sws_t *sws, value *out_vect, value *tmp) {
int i, offset = 0;
intnat out_size = 0;
uint8_t *data;

for (i = 0; i < sws->out.nb_planes; i++)
out_size += sws->out.plane_sizes[i];

out_size += 16;

*out_vect = caml_alloc_tuple(2);
Store_field(
*out_vect, 0,
caml_ba_alloc(CAML_BA_C_LAYOUT | CAML_BA_UINT8, 1, NULL, &out_size));
Store_field(*out_vect, 1, caml_alloc_tuple(sws->out.nb_planes));

data = Caml_ba_data_val(Field(*out_vect, 0));

for (i = 0; i < sws->out.nb_planes; i++) {
*tmp = caml_alloc_tuple(2);
Store_field(*tmp, 0, Val_int(sws->out.plane_sizes[i]));
Store_field(*tmp, 1, Val_int(sws->out.stride[i]));

Store_field(Field(*out_vect, 1), i, *tmp);

sws->out.slice[i] = data + offset;
offset += sws->out.plane_sizes[i];
}

return 0;
}

CAMLprim value ocaml_swscale_convert(value _sws, value _in_vector) {
CAMLparam2(_sws, _in_vector);
CAMLlocal2(out_vect, tmp);
Expand Down Expand Up @@ -436,6 +489,8 @@ CAMLprim value ocaml_swscale_create(value flags_, value in_vector_kind_,
} else if (in_vector_kind == Str) {
sws->get_in_pixels = get_in_pixels_string;
sws->in.owns_data = 1;
} else if (in_vector_kind == PackedBa) {
sws->get_in_pixels = get_in_pixels_packed_ba;
} else {
sws->get_in_pixels = get_in_pixels_ba;
}
Expand All @@ -446,6 +501,8 @@ CAMLprim value ocaml_swscale_create(value flags_, value in_vector_kind_,
sws->alloc_out = alloc_out_string;
sws->copy_out = copy_out_string;
sws->out.owns_data = 1;
} else if (out_vect_kind == PackedBa) {
sws->alloc_out = alloc_out_packed_ba;
} else {
sws->alloc_out = alloc_out_ba;
}
Expand Down

0 comments on commit da98226

Please sign in to comment.