Not logged in.  Login/Logout/Register | List snippets | | Create snippet | Upload image | Upload data

277
LINES

< > BotCompany Repo | #1033823 // JavaCPP ffpmeg Encode H264 Demo [OK on Linux x64]

JavaX source code (desktop) [tags: use-pretranspiled] - run with: x30.jar

Download Jar. Uses 16081K of libraries. Click here for Pure Java version (319L/3K).

!7

// from https://github.com/vzhn/ffmpeg-java-samples/blob/master/src/main/java/EncodeAndMuxH264.java

/*lib 1400538
lib 1400539
lib 1400540
lib 1400541
lib 1400542
lib 1400543*/
lib 1400544 // javacpp ffmpeg examples for linux x64

p { EncodeAndMuxH264.main(args); }

import org.apache.commons.cli.CommandLine;
import org.apache.commons.cli.DefaultParser;
import org.apache.commons.cli.Option;
import org.apache.commons.cli.Options;
import org.apache.commons.cli.HelpFormatter;
import org.apache.commons.cli.ParseException;
import org.bytedeco.javacpp.*;

import java.awt.*;
import java.awt.image.BufferedImage;
import java.awt.image.DataBufferByte;

import static org.bytedeco.javacpp.avcodec.*;
import static org.bytedeco.javacpp.avformat.*;
import static org.bytedeco.javacpp.avutil.*;
import static org.bytedeco.javacpp.swscale.sws_getContext;

sclass EncodeAndMuxH264 {
    private final static String DEFAULT_FPS = "30";
    private static final String DEFAULT_BITRATE = "400000";
    private static final String DEFAULT_WIDTH = "640";
    private static final String DEFAULT_HEIGHT = "320";
    private static final String DEFAULT_GOP = "60";
    private static final String DEFAULT_MAX_B_FRAMES = "12";
    private static final String DEFAULT_N_FRAMES = "300";
    private static final String DEFAULT_PROFILE = "baseline";
    private static final String DEFAULT_FILE = "out.mkv";

    private AVFrame frame;
    private AVFrame rgbFrame;
    private swscale.SwsContext swsContext;
    private BufferedImage image;
    private AVCodecContext cc;
    private int fps;
    private int bitrate;
    private int width;
    private int height;
    private int gopSize;
    private int maxBFrames;
    private int nFrames;
    private String profile;
    private AVCodec codec;
    private AVFormatContext oc;
    private AVOutputFormat fmt;
    private String ofile;
    private AVRational streamTimebase;
    private AVRational codecTimebase;
    private AVPacket pkt;

    private EncodeAndMuxH264() {}

    public static void main(String... argv) throws ParseException {
        Options options = new Options();
        options.addOption("help", false, "show help and exit");
        options.addOption("fps", true, "fps");
        options.addOption("bitrate", true, "bitrate");
        options.addOption("width", true, "width");
        options.addOption("height", true, "height");
        options.addOption("gop", true, "gop");
        options.addOption("max_b_frames", true, "max_b_frames");
        options.addOption("n_frames", true, "number of frames");
        options.addOption("profile", true, "h264 profile");
        options.addOption("file", true, "output file name");
        CommandLine cmd = new DefaultParser().parse(options, argv);
        if (cmd.hasOption("help")) {
            HelpFormatter helpFormatter = new HelpFormatter();
            helpFormatter.printHelp("EncodeAndMuxH264 [options]", options);
        } else {
            System.out.println("options:");
            EncodeAndMuxH264 instance = new EncodeAndMuxH264();
            instance.fps = Integer.parseInt(getOption(cmd, "fps", DEFAULT_FPS));
            instance.bitrate = Integer.parseInt(getOption(cmd, "bitrate", DEFAULT_BITRATE));
            instance.width = Integer.parseInt(getOption(cmd,"width", DEFAULT_WIDTH));
            instance.height = Integer.parseInt(getOption(cmd,"height", DEFAULT_HEIGHT));
            instance.gopSize = Integer.parseInt(getOption(cmd,"gop", DEFAULT_GOP));
            instance.maxBFrames = Integer.parseInt(getOption(cmd,"max_b_frames", DEFAULT_MAX_B_FRAMES));
            instance.nFrames = Integer.parseInt(getOption(cmd,"n_frames", DEFAULT_N_FRAMES));
            instance.profile = getOption(cmd,"profile", DEFAULT_PROFILE);
            instance.ofile = getOption(cmd,"file", DEFAULT_FILE);

            instance.start();
        }
    }

    private static String getOption(CommandLine cmd, String key, String defaultValue) {
        String v = cmd.getOptionValue(key, defaultValue);
        System.out.println("\t" + key + " = \"" + v + "\"");
        return v;
    }

    private void start() {
        allocCodecContext();

        AVPacket pkt = av_packet_alloc();

        allocFrame(cc);
        allocRgbFrame(cc);
        allocSwsContext();
        allocOutputContext();

        encodeVideo(pkt);
        writeDelayedFrames();

        av_write_trailer(oc);
        free(cc, oc);
    }

    private void writeDelayedFrames() {
        sendFrame(null);
    }

    private void encodeVideo(AVPacket pkt) {
        for (int i = 0; i < nFrames; i++) {
            frame.pts(avutil.av_rescale_q(i, codecTimebase, streamTimebase));

            drawFrame(i);
            sendFrame(frame);
        }
    }

    private void sendFrame(AVFrame o) {
        int r = avcodec.avcodec_send_frame(cc, o);
        if (r == 0) {
            receivePacket();
        } else {
            throw new RuntimeException("error: " + r);
        }
    }

    private void drawFrame(int n) {
        Graphics gc = image.getGraphics();
        gc.clearRect(0, 0, image.getWidth(), image.getHeight());
        gc.setFont(gc.getFont().deriveFont(50f));
        gc.drawString(String.format("pts: %d", n), 200, 200);
        gc.dispose();

        DataBufferByte dataBufferByte = (DataBufferByte) image.getRaster().getDataBuffer();
        rgbFrame.data(0).put(dataBufferByte.getData());

        swscale.sws_scale(
            swsContext, rgbFrame.data(), rgbFrame.linesize(), 0,
            frame.height(), frame.data(), frame.linesize()
        );
    }

    private void allocOutputContext() {
        oc = new AVFormatContext();
        pkt = new AVPacket();
        int r = avformat_alloc_output_context2(oc, null, null, ofile);
        if (r < 0) {
            throw new RuntimeException("could not allocate output context");
        }
        fmt = oc.oformat();
        AVStream st = avformat_new_stream(oc, codec);
        avcodec_parameters_from_context(st.codecpar(), cc);
        st.time_base(cc.time_base());

        av_dump_format(oc, 0, ofile, 1);

        /* open the output file, if needed */
        PointerPointer pp = new PointerPointer(1);
        try {
            if (avio_open(pp, new BytePointer(ofile), AVIO_FLAG_WRITE) <0){
                throw new RuntimeException("Could not open " + fmt);
            }
            oc.pb(new AVIOContext(pp.get()));
        } finally {
            pp.deallocate();
        }

        /* Write the stream header, if any. */
        if (avformat_write_header(oc, (AVDictionary) null) < 0) {
            throw new RuntimeException("Error occurred when opening output file\n");
        }

        streamTimebase = st.time_base();
    }

    private void allocCodecContext() {
        codecTimebase = new avutil.AVRational();
        codecTimebase.num(1);
        codecTimebase.den(fps);
        codec = avcodec_find_encoder(AV_CODEC_ID_H264);
        cc = avcodec_alloc_context3(codec);

        cc.bit_rate(bitrate);
        cc.width(width);
        cc.height(height);
        cc.time_base(codecTimebase);
        cc.gop_size(gopSize);
        cc.max_b_frames(maxBFrames);
        if (profile != null && !"".equals(profile)) {
            av_opt_set(cc.priv_data(), "profile", profile, 0);
        }

        cc.pix_fmt(avutil.AV_PIX_FMT_YUV420P);
        cc.flags(cc.flags() | AV_CODEC_FLAG_GLOBAL_HEADER);
        if (avcodec_open2(cc, codec, (AVDictionary) null) < 0) {
            throw new RuntimeException("could not open codec");
        }
    }

    private void free(AVCodecContext cc, AVFormatContext oc) {
        avcodec_close(cc);
        avcodec_free_context(cc);
        av_free(rgbFrame.data(0));
        av_free(frame.data(0));
        av_free(rgbFrame);
        av_free(frame);

        avio_close(oc.pb());
        av_free(oc);
    }

    private void allocSwsContext() {
        swsContext = sws_getContext(rgbFrame.width(), rgbFrame.height(), rgbFrame.format(),
                frame.width(), frame.height(), frame.format(), swscale.SWS_BICUBIC,
                null, null, (DoublePointer) null);

        if (swsContext.isNull()) {
            throw new RuntimeException("Could not init sws context!");
        }
    }

    private void allocRgbFrame(AVCodecContext cc) {
        image = new BufferedImage(cc.width(), cc.height(), BufferedImage.TYPE_3BYTE_BGR);

        rgbFrame = av_frame_alloc();
        rgbFrame.format(AV_PIX_FMT_BGR24);
        rgbFrame.width(cc.width());
        rgbFrame.height(cc.height());
        int ret = av_frame_get_buffer(rgbFrame, 32);
        if (ret < 0) {
            throw new RuntimeException("Could not allocate the video frame data");
        }
    }

    private void allocFrame(AVCodecContext cc) {
        frame = av_frame_alloc();
        frame.format(cc.pix_fmt());
        frame.width(cc.width());
        frame.height(cc.height());
        int ret = av_frame_get_buffer(frame, 32);
        if (ret < 0) {
            throw new RuntimeException("Could not allocate the video frame data");
        }
    }

    private void receivePacket() {
        int r;
        while ((r = avcodec.avcodec_receive_packet(cc, pkt)) == 0) {
            r = av_interleaved_write_frame(oc, pkt);
            av_packet_unref(pkt);
            if (r != 0) {
                throw new RuntimeException("Error while writing video frame\n");
            }
        }

        if (r != AVERROR_EAGAIN() && r != AVERROR_EOF()) {
            throw new RuntimeException("error");
        }
    }
}

Author comment

Began life as a copy of #1033821

download  show line numbers  debug dex  old transpilations   

Travelled to 3 computer(s): bhatertpkbcr, mowyntqkapby, mqqgnosmbjvj

No comments. add comment

Snippet ID: #1033823
Snippet name: JavaCPP ffpmeg Encode H264 Demo [OK on Linux x64]
Eternal ID of this version: #1033823/4
Text MD5: ca049d66e323f8146d8a17ec126aa839
Transpilation MD5: c5d08aceccbd18027d08e4fe5bcec436
Author: stefan
Category: javax / gazelle v
Type: JavaX source code (desktop)
Public (visible to everyone): Yes
Archived (hidden from active list): No
Created/modified: 2022-01-06 19:17:12
Source code size: 9795 bytes / 277 lines
Pitched / IR pitched: No / No
Views / Downloads: 93 / 581
Version history: 3 change(s)
Referenced in: [show references]