48

I want to join two mp4 files to create a single one. The video streams are encoded in h264 and the audio in aac. I can not re-encode the videos to another format due to computational reasons. Also, I cannot use any GUI programs, all processing must be performed with Linux command line utilities. FFmpeg cannot do this for mpeg4 files so instead I used MP4Box:

MP4Box -add video1.mp4 -cat video2.mp4 newvideo.mp4

Unfortunately the audio gets all mixed up. I thought that the problem was that the audio was in aac so I transcoded it in mp3 and used again MP4Box. In this case the audio is fine for the first half of newvideo.mp4 (corresponding to video1.mp4) but then their is no audio and I cannot navigate in the video also.

My next thought was that the audio and video streams had some small discrepancies in their lengths that I should fix. So for each input video I splitted the video and audio streams and then joined them with the -shortest option in FFmpeg.

Thus for the first video I ran:

 avconv -y -i video1.mp4 -c copy -map 0:0 videostream1.mp4
 avconv -y -i video1.mp4 -c copy -map 0:1 audiostream1.m4a
 avconv -y -i videostream1.mp4 -i audiostream1.m4a  -c copy -shortest  video1_aligned.mp4

Similarly for the second video and then used MP4Box as previously. Unfortunately this didn't work either. The only success I had was when I joined the video streams separately (i.e. videostream1.mp4 and videostream2.mp4) and the audio streams (i.e. audiostream1.m4a and audiostream2.m4a) and then joined the video and audio in a final file. However, the synchronization is lost for the second half of the video. Concretely, there is a 1 sec delay of audio and video. Any suggestions are really welcome.

llogan
  • 63,280

7 Answers7

63

The best way to do this currently is with the concat demuxer. First, create a file called inputs.txt formatted like so:

file '/path/to/input1.mp4'
file '/path/to/input2.mp4'
file '/path/to/input3.mp4'

Then, simply run this ffmpeg command:

ffmpeg -f concat -i inputs.txt -c copy output.mp4

See also concatenation in ffmpeg FAQ.


I'm keeping the following here for the benefit of anyone using older versions of ffmpeg.

The latest versions of ffmpeg can do this: you'll have to remux the files into mpeg transport streams first (fairly processor-light, as it's only changing the container format):

ffmpeg -i input.mp4 -map 0 -c copy -f mpegts -bsf h264_mp4toannexb middle.ts

If that throws up an error [1] about h264, you may need to use:

ffmpeg -i input.mp4 -map 0 -c copy -f mpegts -bsf:v h264_mp4toannexb middle.ts

You'll have to do this separately with each input file. To concatenate the files together, use:

ffmpeg -i "concat:middle1.ts|middle2.ts|middle3.ts" -c copy output.mp4

If that throws up an error about aac, you may need to use

ffmpeg -i "concat:middle1.ts|middle2.ts|middle3.ts" -c copy -absf aac_adtstoasc output.mp4

If your system supports named pipes, you can do this without creating intermediate files.

mkfifo temp0 temp1

You'll have to do the following in three separate virtual terminals:

ffmpeg -i input0.mp4 -map 0 -c copy -f mpegts -bsf h264_mp4toannexb -y temp0
ffmpeg -i input1.mp4 -map 0 -c copy -f mpegts -bsf h264_mp4toannexb -y temp1
ffmpeg -f mpegts -i "concat:temp0|temp1" -c copy -absf aac_adtstoasc output.mp4

If output.mp4 already exists, the third ffmpeg will ask you whether you want to overwrite it, but it will do this after it has accessed the FIFOs, and this will make the first to ffmpegs close. So make sure that you choose an unused name for your output file.

This may not work if your input files are different - I believe that differences in bit rate are OK, but frame size, frame rate etc have to match up.

evilsoup
  • 14,056
8

Here is a one-liner that makes it all that much easier. It does everything needed to be done in order to finally provide you with one resulting video file. Cheers!

find *.mp4 | sed 's:\ :\\\ :g'| sed 's/^/file /' > list.txt; ffmpeg -f concat -i list.txt -c copy output.mp4; rm list.txts
Anonymous
  • 189
8

For mp4 the only working solution I found was with MP4Box from gpac package

#!/bin/bash
filesList=""
for file in $(ls *.mp4|sort -n);do
    filesList="$filesList -cat $file"
done
MP4Box $filesList -new merged_files_$(date +%Y%m%d_%H%M%S).mp4

or command is

MP4Box -cat file1.mp4 -cat file2.mp4 -new mergedFile.mp4

with mencoder and avconv I could'nt make it work :-(

3

On a Mac (macOS 10.13 High Sierra) I used the following with success:

for f in *.m4a; do echo "file '$f'" >> mylist.txt; done

Reorder as above similarly if necessary (note that the ./ was removed - the path was causing ffmpeg to throw the unsafe file name error).

Then I had to remove the codec and bitrate specified for the default MacPorts ffmpeg to like it:

ffmpeg -f concat -i mylist.txt  output.m4a
2

Thanks for the script dvo. It was a great starting point for me. I had no problem with using named pipes so I adapted the script to use them instead. Plus I fixed it to work with filenames with spaces in them. Output to named pipes does not "go" until you begin reading from them, hence the need to background the initial remux tasks with an & before the final call to avconv. Plus, don't forget to use -c copy, or you end up re-encoding the whole lot.

#!/bin/bash
for FILE in "$@"; do
   TAG=$(echo "$FILE" | sed -e s/[^A-Za-z0-9.]/_/g)
   mkfifo $TAG.mp4concatpipe
   avconv -loglevel "quiet" -i "$FILE"  -f mpegts -c copy -bsf h264_mp4toannexb -y $TAG.mp4concatpipe &
   IN=$IN\|$TAG.fifo
done

IN=${IN#\|}
avconv -i concat:$IN -c copy out.mp4
rm *.mp4concatpipe
Mark
  • 21
1

Thanks evilsoup; here is a little script to automate the process, using the more recent avconv.
BTW, named pipes did not work for me (output to them got stuck).

   #!/bin/bash

    for FILE in $@; do
      avconv -i $FILE -map 0 -f mpegts -bsf h264_mp4toannexb -c:v copy -y $FILE.tmp
      IN=$IN\|$FILE.tmp
    done

    IN=${IN#\|}
    avconv -i concat:"$IN" out.mp4

    for FILE in $@; do
      rm $FILE.tmp
    done
dvo
  • 155
1

Have enjoyed great success with

for f in ./*.m4a; do echo "file '$f'" >> mylist.txt; done

you may need to reorder mylist.txt slightly

ffmpeg -f concat -i mylist.txt -c:a libfdk_aac -b:a 128k output.m4a

ffmpeg version 2.2.3