-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathtranscoding.ml
99 lines (85 loc) · 2.97 KB
/
transcoding.ml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
let () = Printexc.record_backtrace true
let () =
if Array.length Sys.argv < 3 then (
Printf.printf "usage: %s input_file output_file\n" Sys.argv.(0);
exit 1);
Avutil.Log.set_level `Debug;
Avutil.Log.set_callback print_string;
let src = Av.open_input Sys.argv.(1) in
let dst = Av.open_output Sys.argv.(2) in
let audio_codec = Avcodec.Audio.find_encoder_by_name "aac" in
let video_codec = Avcodec.Video.find_encoder_by_name "mpeg4" in
let iass = Av.get_audio_streams src in
let oass =
iass
|> List.map (fun (i, _, params) ->
let channel_layout = Avcodec.Audio.get_channel_layout params in
let sample_format = Avcodec.Audio.get_sample_format params in
let sample_rate = Avcodec.Audio.get_sample_rate params in
let time_base = { Avutil.num = 1; den = sample_rate } in
( i,
Av.new_audio_stream ~channel_layout ~sample_format ~sample_rate
~time_base ~codec:audio_codec dst ))
in
let frame_rate = { Avutil.num = 25; den = 1 } in
let time_base = { Avutil.num = 1; den = 25 } in
let ivss = Av.get_video_streams src in
let ovss =
ivss
|> List.map (fun (i, _, params) ->
let width = Avcodec.Video.get_width params in
let height = Avcodec.Video.get_height params in
let pixel_format =
match Avcodec.Video.get_pixel_format params with
| None -> failwith "Pixel format unknown!"
| Some f -> f
in
( i,
Av.new_video_stream ~pixel_format ~frame_rate ~time_base ~width
~height ~codec:video_codec dst ))
in
let isss = Av.get_subtitle_streams src in
let osss =
isss
|> List.map (fun (i, _, params) ->
let codec =
Avcodec.Subtitle.find_encoder
(Avcodec.Subtitle.get_params_id params)
in
(i, Av.new_subtitle_stream ~time_base ~codec dst))
in
let process_frame frame =
let metadata = Avutil.Frame.metadata frame in
List.iter
(fun (key, value) -> Printf.printf "Frame metadata: %s->%s\n%!" key value)
metadata;
Avutil.Frame.set_metadata frame (("encoder", "ocaml-ffmpeg") :: metadata)
in
let rec f () =
match
Av.read_input
~audio_frame:(List.map (fun (_, s, _) -> s) iass)
~video_frame:(List.map (fun (_, s, _) -> s) ivss)
~subtitle_frame:(List.map (fun (_, s, _) -> s) isss)
src
with
| `Audio_frame (i, frame) ->
process_frame frame;
Av.write_frame (List.assoc i oass) frame;
f ()
| `Video_frame (i, frame) ->
process_frame frame;
Av.write_frame (List.assoc i ovss) frame;
f ()
| `Subtitle_frame (i, frame) ->
process_frame frame;
Av.write_frame (List.assoc i osss) frame;
f ()
| exception Avutil.Error `Eof -> ()
| _ -> assert false
in
f ();
Av.close src;
Av.close dst;
Gc.full_major ();
Gc.full_major ()