-
Notifications
You must be signed in to change notification settings - Fork 913
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
ffmpeg 3.1 supports h264 hardware assist encoding #365
Comments
And would that hardware assisted encoder be automatically used by motion when creating movies? |
I managed to compile ffmpeg 3.1 with the options you have suggested and when specifying "mp4" as ffmpeg_video_codec to Mr Dave's motion version, it creates h264-encoded movies. However the movies have wrong frame rates (10k or higher!) and are therefore unplayable. |
Hi, Can you run this command with ffmpeg-3.1: On July 3, 2016 at 9:04:53 AM, Calin Crisan (notifications@github.com) I managed to compile ffmpeg 3.1 with the options you have suggested and — |
Which test.mp4? Can you give me a link where I can download the file from? |
The command generates test.mp4 On July 3, 2016 at 9:15:40 AM, Calin Crisan (notifications@github.com) Which test.mp4? Can you give me a link where I can download the file from? — |
My apologies, I didn't read your command like carefully enough. Yes, it works and it creates a file with the correct frame rate. However when invoked from motion (through libavcodec I suppose) the movie has a wrong frame rate set. |
Fine, I will let you play with motion/ffmpeg-3.1 for a while. In the mean time, please post a pointer to the version of Last but not least there is now a ffmpeg-3.1.1. The guys Enjoy! On July 3, 2016 at 9:41:22 AM, Calin Crisan (notifications@github.com) My apologies, I didn't read your command like carefully enough. Yes, it — |
Actually I've been playing with ffmpeg-3.1.1. The motion version I'm using is the latest commit of Mr Dave's fork. |
It turns out the framerate problem has nothing to do with enabling h264 hardware accelerated encoding. I disabled it and went the software libx264 way. The framerate problem is still there. However with older ffmpeg versions (2.8.x) I don't seem to be having this problem. |
Cool, just report it to the ffmpeg bug tracker: http://ffmpeg.org/bugreports.html Cheers On July 3, 2016 at 12:01:53 PM, Calin Crisan (notifications@github.com) It turns out the framerate problem has nothing to do with enabling h264 — |
Hi Not sure if you solved the problem or not. If motion does not work , it is If you are using an old version of ffmpeg it should not be too Those minor changes can be easily discerned by looking Last but not least a static binary release of ffmpeg-3.1.1/ffmpeg Amancio
|
@ccrisan Have you played around with motion + ffmpeg 3.x combination? What is the status? |
@jasaw I'll look into it. |
Looks like to support hwaccel on the Pi, we need to add "--enable-omx --enable-omx-rpi --enable-mmal" to ffmpeg config via the buildroot xconfig. Also need to enable "bellagio" package, because ffmpeg omx depends on libomxil-bellagio.so. Buildroot at this point will complain about both rpi-userland and bellagio both providing libopenmax. I just removed libopenmax from rpi-userland Config.in and mk file just to proceed with buildroot compilation. I haven't got time to test the generated image yet. |
hey @jasaw @ccrisan, tried this before with @kerberos-io, everything compiled properly but when executing the ffmpeg command it was complaining about a couple of .so missing. What I did is moved them from /usr/bin to /opt/vc, this made the complaining going away but caused ffmpeg to crash when selecting the h264_omx encoder. At the moment we are implementing OpenMAX directly into Kerberos.io/machiner, and we are able to stream MJPEG at 25FPS at 1280x720 and record at 30 FPS 1280x720, and 90 FPS at 640x480. Maybe integrating OpenMax out-of-the-box can also be a solution for motion. |
I have managed to build ffmpeg with
However, when I run Any ideas? |
@ccrisan I didn't mean that it crashed at compiling. It did crash if I used the h264_omx codec on command line. What happens if you write |
I managed to get ffmpeg to encode with hwaccel in isolation. This is very much a hack, but here's what I've done:
Next thing is to get motion to use hwaccel ffmpeg. |
@cedricve my |
For the record, I ran both versions of motionEyeOS in parallel (on PI1 boards) and did not see any significant difference between the two. I set them up to encode h264 movies continuously via our good old motion. Either my hardware-accelerated ffmpeg isn't hardware accelerated at all or motion does not use it. I really don't know too much about this hardware accelerated stuff but anyways, these are my findings. |
Pretty sure motion doesn't use it, you should see the difference definitely (either in frame rate and CPU usage). |
Alright... I managed to get motion to record movie using ffmpeg with h264_omx encoder. I got 18 fps, whereas software encoder gave me 2 fps, but the h264_omx bitrate is not great because I haven't figured out how to get motion to set bitrate via ffmpeg's API. Anyway, this is the interesting bit in my motion.log:
Here's what I've done. Again, this is all hacked up to prove that it can be done.
The proper way of supporting this is to get motion to specify which encoder to use when setting up a new movie. Motion current tells ffmpeg the output file extension is "mp4", then ffmpeg guesses which encoder to use. |
Did another quick hack to improve the h264 video quality by setting the CRF (Constant Rate Factor). At best CRF quality, 800 x 600 resolution, I'm doing 25 fps 40% CPU load on a Pi Zero W. Woohoo ! This will do for me. The CRF hack that I did, if anyone is interested. |
@ccrisan I tried using rpi-userland's libopenmax but ffmpeg doesn't work with that. Looks like it needs bellagio's libopenmax, unless I made a mistake. You said you managed to get ffmpeg to use rpi-userland's libopenmax, do you mind sharing more details so I can replicate it? I had a quick look into buildroot on how to enable bellagio package "properly", but not sure how to do it. Basically, both rpi-userland and bellagio pakages are providing libopenmax virtual package, and buildroot takes that as configuration error and doesn't proceed to compilation. Any ideas on how we should go about getting ffmpeg omx to work? ffmpeg should have mpeg4_omx support as well, but not present on the version used by motioneyeos. |
@jasaw I simply added the following extra configure options to ffmpeg:
Having it look for headers in |
@ccrisan That's what I did as well, and managed to get ffmpeg to compile with rpi-userland headers, but when I ran ffmpeg, I get this:
I didn't get this error when I used bellagio's libopenmax. I don't know what the differences are between the rpi-userland and bellagio. |
indeed @jasaw Ttat was exactly what I mean with previous comment. |
Apparently, there are 3 versions of libopenmaxil.so
Either number 2 or 3 is normally installed at /opt/vc/lib directory. ffmpeg seems to work when I use the /usr/lib version, but doesn't work with the rpi-firmware versions. I don't know why yet. Anyway, there is no need to use bellagio's libopenmax. |
I've created pull request 998 as my first attempt to address this issue. |
It appears that ffmpeg occasionally locks up when encoding certain resolutions. I've been monitoring GPU memory usage and it seems to be doing OK (no OOM problem). CPU usage is low too, never reach full throttle. Anyone has any clue to what might be happening? Note: I'm running ffmpeg version 3.2.3 with h264_omx encoder, latest motion master (git hash 280141f4178f2d656a2af4c5b3d3c430f41646a1) with hw_accel patch. I'm seeing this problem on Pi B, Pi2 B, and Pi Zero W. Haven't tested with other Pis. Edit: ffmpeg version 3.3.2 (latest release) has the same issue. |
After fixing recorded movie playback bug on motion, I'm finally able to really test the performance of ffmpeg h264_omx encoder. At 800 x 600 resolution, all models of Raspberry Pis have no issue running higher than 20 fps but at 1920 x 1080 resolution, I only managed to get 3 fps on a Pi2 B. From what I've read, the bottleneck appears to be memory bandwidth between ARM and GPU. With motion software, we currently have this pipeline: @cedricve You mentioned you're able to achieve high frame rates on a Pi.
Would you be able to share with us what kind of set up you have to get these frame rates? I only know your software is talking to OpenMax directly. What was your pipeline configuration? Pi camera --> ARM CPU processing --> GPU encoding --> ARM CPU file storage? @ccrisan Have you tried getting |
Looks like the avcodec_send_frame problem is actually caused by OMX_EmptyThisBuffer locking up, and that's in the Raspberry Pi GPU firmware territory. |
@jasaw we've seen this also in the Kerberos.io project, when recording at a higher resolution we get locks and Kerberos.io freezes. I confirm having this myself after 9days running 24/7 at a resolution of 1280x960. This might be caused because the os isn't able to read frames fast enough, did you tried to lower FPS on the higher resolution? |
@cedricve Yes, I've tried with 3 fps 1280 x 720, and it's still locking up. Weird thing is, I haven't seen 30 fps 1920 x 1080 lock up, even though it's only able to encode 5 fps. Although I haven't tested 1920 x 1080 for more than 1 day. With 1280 x 720, it locks up very quickly, easy to reproduce, that's why I'm using it to debug. |
Hmm well this is a different use case then. Although we're using OMX directly, not in combo with ffmpeg. Not sure how many FPS you can achieve with a RPI Camera on motioneyeos? @ccrisan can you answer? I'm trying to integrate your PR into KiOS and see if I can simulate with ip cameras. My approach will be different: IP camera -> OpenCV -> extract frames and do images processing -> when motion fireup an ffmpeg command with the h264_omx encoder specified. |
@cedricve That's a broad question. The best that I have seen on a RPi 3 at the lowest resolution possible is somewhere about 20fps. With the current motionEyeOS setup you're obviously bound by the CPU as the hardware acceleration is not used at all. |
Having motion use openmax when talking to the PI cam either via ffmpeg or directly, would be a great thing to have. I believe it's worth everyone's time and effort, a lot more than with any other motionEye features. |
@cedricve I have thought about modifying motion to talk to OpenMAX directly, but wanted to get a better understanding of what the current problems are, where the bottlenecks are, in the hope that our implementation can avoid those issues. My other concern is code maintenance: ideally, we do a pull-request to merge into Mr-Dave's motion master so other developers can help maintain the code. Let's discuss further. I'll email you directly. |
I tried running motion with extpipe to see what is causing the lock up, and this is what I found. Encode via extpipe Running motion with extpipe to ffmpeg h264_omx is stable at various resolutions. This configuration also gave me higher frame rate:
extpipe config:
Encode via C API Running motion encoding via ffmpeg C API hangs at 1280 x 720 resolution. Exact same encoder configuration (bitrate, profile, etc...) as extpipe version. This configuration gave me lower frame rate:
At this stage, I can only conclude that motion is doing something wrong.
@cedricve Your use case should be OK. |
Hardware accelerated video encoding is now supported in the pre-release 20171008. |
@cedricve The lockup only happens when passing a GPU/ARM shared buffer back to the GPU with zero copy. Zero copy is the important bit here. To enable zero copy, the resolution width must be divisible by 32 and height divisible by 16, so image can be handed over to GPU without copying. Resolutions 640x480 and 1280x720 both are divisible by 32 width and 16 height, so you get a lockup. To workaround zero copy lockup, you'll need to do a copy regardless of the resolution. 6by9 thinks it's caused by the Linux kernel memory mapping/management. I am reluctant to dive deep into the kernel memory management code. I've reported this issue here: raspberrypi/firmware#851. I hacked up motion software to support zero copy, i.e. taking images straight from MMAL camera (height & width divisible by 32 &16), and let motion do its thing, and pass the images to OMX encoder, all with zero copy. Curiously, it sort of works on my Raspbian, but locks up on motionEyeOS. I guess both devices had different versions of kernel at that time. It "sort of works" because it was encoding at 1920x1088 resolution 30 fps for a few seconds, and paused for a few seconds, and continued at 30 fps. I didn't have time to investigate further and gave up on making motion zero copy because it requires too much code changes and it's impossible to get it merged upstream. |
I'm trying to use ffmpeg with
Can someone advise as to which OMX headers are required to build a working binary? |
Make sure you have the right version of libraries. |
Where the lib in /usr come from? I don't see that on raspbian, only the files in /opt/vc |
I upgraded from jessie to stretch and it fixed the issue. |
@jasaw the patch file you provided above appears to not work with the latest source. Can you tell me which checkin/version of the source that patch should work with? I really would like to be able to enable crf with the hardware encoder. Thank you!! |
@cmsmith81 Just FYI, h264_omx encoder only accepts bitrate option, not crf option. If you are trying to enable omx on your own project, check out the latest motion master branch. All my motion patches have been merged upstream. You'll also need to apply this patch to enable it. |
@jasaw thanks so much for your reply! I'll check out the code - - I'm needing this for a different t project, but your code will probably show me what I need. I'm needing to encode true lossless h264 on a Raspberry PI, which needs the "crf=0" option. |
@jasaw could you please be so kind and see if the latest nightly build (dev20181126) makes use of the h264 OMX acceleration? I've tried to bring thingOS/motionEyeOS as closely as possible to BuildRoot and the |
@ccrisan I've tested H264 OMX hw accel with dev20181126 and it's still working as expected. |
@jasaw thanks. Then we're ready for a new release. |
to compile ffmpeg with hardware assist h264 encoding:
./configure --enable-mmal --enable-omx-rpi --enable-omx
if you have an RPI3 , you can use multiple cores to speed up compilation:
make -j4
to test:
./ffmpeg -f v4l2 -input_format yuv420p -framerate 25 -video_size 640x480 -i /dev/video0 -frames 500 -an -c:v h264_omx test.mp4
ffmpeg will generate warnings just ignore them for now.
The omx h264 hardware assist encoding was ported from libav by Aman Gupta.
Amancio
The text was updated successfully, but these errors were encountered: