2022-04-06 17:46:41 +02:00
|
|
|
package ffmpeg
|
|
|
|
|
|
|
|
import (
|
2022-04-13 22:17:59 +02:00
|
|
|
"fmt"
|
2022-04-06 17:46:41 +02:00
|
|
|
"os/exec"
|
2022-04-13 22:17:59 +02:00
|
|
|
|
|
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
2022-04-06 17:46:41 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// AvcConvertCommand returns the command for converting video files to MPEG-4 AVC.
|
|
|
|
func AvcConvertCommand(fileName, avcName, ffmpegBin, bitrate string, encoder AvcEncoder) (result *exec.Cmd, useMutex bool, err error) {
|
2022-04-13 22:17:59 +02:00
|
|
|
if fileName == "" {
|
|
|
|
return nil, false, fmt.Errorf("empty input filename")
|
|
|
|
} else if avcName == "" {
|
|
|
|
return nil, false, fmt.Errorf("empty output filename")
|
|
|
|
}
|
|
|
|
|
2022-04-06 17:46:41 +02:00
|
|
|
// Don't transcode more than one video at the same time.
|
|
|
|
useMutex = true
|
|
|
|
|
2023-02-22 16:33:33 +01:00
|
|
|
// Don't use hardware transcoding for animated images.
|
|
|
|
if fs.TypeAnimated[fs.FileType(fileName)] != "" {
|
2022-04-13 22:17:59 +02:00
|
|
|
result = exec.Command(
|
|
|
|
ffmpegBin,
|
|
|
|
"-i", fileName,
|
|
|
|
"-movflags", "faststart",
|
|
|
|
"-pix_fmt", "yuv420p",
|
|
|
|
"-vf", "scale=trunc(iw/2)*2:trunc(ih/2)*2",
|
|
|
|
"-f", "mp4",
|
|
|
|
"-y",
|
|
|
|
avcName,
|
|
|
|
)
|
|
|
|
|
|
|
|
return result, useMutex, nil
|
|
|
|
}
|
2022-04-06 17:46:41 +02:00
|
|
|
|
|
|
|
// Display encoder info.
|
|
|
|
if encoder != SoftwareEncoder {
|
2022-04-13 22:17:59 +02:00
|
|
|
log.Infof("convert: ffmpeg encoder %s selected", string(encoder))
|
2022-04-06 17:46:41 +02:00
|
|
|
}
|
|
|
|
|
2022-04-13 22:17:59 +02:00
|
|
|
switch encoder {
|
|
|
|
case IntelEncoder:
|
|
|
|
// ffmpeg -hide_banner -h encoder=h264_qsv
|
2022-04-06 17:46:41 +02:00
|
|
|
format := "format=rgb32"
|
|
|
|
result = exec.Command(
|
|
|
|
ffmpegBin,
|
|
|
|
"-qsv_device", "/dev/dri/renderD128",
|
|
|
|
"-i", fileName,
|
|
|
|
"-c:a", "aac",
|
|
|
|
"-vf", format,
|
|
|
|
"-c:v", string(encoder),
|
2023-03-02 03:30:45 +01:00
|
|
|
"-map", "0:v:0",
|
2022-04-06 17:46:41 +02:00
|
|
|
"-vsync", "vfr",
|
|
|
|
"-r", "30",
|
|
|
|
"-b:v", bitrate,
|
2022-05-01 18:09:46 +02:00
|
|
|
"-bitrate", bitrate,
|
2022-04-06 17:46:41 +02:00
|
|
|
"-f", "mp4",
|
|
|
|
"-y",
|
|
|
|
avcName,
|
|
|
|
)
|
|
|
|
|
2022-04-13 22:17:59 +02:00
|
|
|
case AppleEncoder:
|
|
|
|
// ffmpeg -hide_banner -h encoder=h264_videotoolbox
|
|
|
|
format := "format=yuv420p"
|
2022-04-06 17:46:41 +02:00
|
|
|
result = exec.Command(
|
|
|
|
ffmpegBin,
|
|
|
|
"-i", fileName,
|
|
|
|
"-c:v", string(encoder),
|
2023-03-02 03:30:45 +01:00
|
|
|
"-map", "0:v:0",
|
2022-04-06 17:46:41 +02:00
|
|
|
"-c:a", "aac",
|
|
|
|
"-vf", format,
|
|
|
|
"-profile", "high",
|
|
|
|
"-level", "51",
|
|
|
|
"-vsync", "vfr",
|
|
|
|
"-r", "30",
|
|
|
|
"-b:v", bitrate,
|
|
|
|
"-f", "mp4",
|
|
|
|
"-y",
|
|
|
|
avcName,
|
|
|
|
)
|
2022-04-13 22:17:59 +02:00
|
|
|
|
2022-09-16 14:14:49 +02:00
|
|
|
case VAAPIEncoder:
|
|
|
|
format := "format=nv12,hwupload"
|
|
|
|
result = exec.Command(
|
|
|
|
ffmpegBin,
|
|
|
|
"-hwaccel", "vaapi",
|
|
|
|
"-i", fileName,
|
|
|
|
"-c:a", "aac",
|
|
|
|
"-vf", format,
|
|
|
|
"-c:v", string(encoder),
|
2023-03-02 03:30:45 +01:00
|
|
|
"-map", "0:v:0",
|
2022-09-16 14:14:49 +02:00
|
|
|
"-vsync", "vfr",
|
|
|
|
"-r", "30",
|
|
|
|
"-b:v", bitrate,
|
|
|
|
"-f", "mp4",
|
|
|
|
"-y",
|
|
|
|
avcName,
|
|
|
|
)
|
|
|
|
|
2022-04-13 22:17:59 +02:00
|
|
|
case NvidiaEncoder:
|
|
|
|
// ffmpeg -hide_banner -h encoder=h264_nvenc
|
2022-04-06 17:46:41 +02:00
|
|
|
result = exec.Command(
|
|
|
|
ffmpegBin,
|
2022-11-03 16:18:04 +01:00
|
|
|
"-hwaccel", "auto",
|
2022-04-06 17:46:41 +02:00
|
|
|
"-i", fileName,
|
|
|
|
"-pix_fmt", "yuv420p",
|
|
|
|
"-c:v", string(encoder),
|
2023-03-02 03:30:45 +01:00
|
|
|
"-map", "0:v:0",
|
2022-04-06 17:46:41 +02:00
|
|
|
"-c:a", "aac",
|
|
|
|
"-preset", "15",
|
|
|
|
"-pixel_format", "yuv420p",
|
|
|
|
"-gpu", "any",
|
|
|
|
"-vf", "format=yuv420p",
|
|
|
|
"-rc:v", "constqp",
|
|
|
|
"-cq", "0",
|
|
|
|
"-tune", "2",
|
2022-06-19 15:32:48 +02:00
|
|
|
"-r", "30",
|
2022-04-06 17:46:41 +02:00
|
|
|
"-b:v", bitrate,
|
|
|
|
"-profile:v", "1",
|
2022-11-03 16:18:04 +01:00
|
|
|
"-level:v", "auto",
|
2022-04-06 17:46:41 +02:00
|
|
|
"-coder:v", "1",
|
|
|
|
"-f", "mp4",
|
|
|
|
"-y",
|
|
|
|
avcName,
|
|
|
|
)
|
|
|
|
|
2022-04-13 22:17:59 +02:00
|
|
|
case Video4LinuxEncoder:
|
|
|
|
// ffmpeg -hide_banner -h encoder=h264_v4l2m2m
|
|
|
|
format := "format=yuv420p"
|
2022-04-06 17:46:41 +02:00
|
|
|
result = exec.Command(
|
|
|
|
ffmpegBin,
|
|
|
|
"-i", fileName,
|
|
|
|
"-c:v", string(encoder),
|
2023-03-02 03:30:45 +01:00
|
|
|
"-map", "0:v:0",
|
2022-04-06 17:46:41 +02:00
|
|
|
"-c:a", "aac",
|
|
|
|
"-vf", format,
|
|
|
|
"-num_output_buffers", "72",
|
|
|
|
"-num_capture_buffers", "64",
|
|
|
|
"-max_muxing_queue_size", "1024",
|
|
|
|
"-crf", "23",
|
|
|
|
"-vsync", "vfr",
|
|
|
|
"-r", "30",
|
|
|
|
"-b:v", bitrate,
|
|
|
|
"-f", "mp4",
|
|
|
|
"-y",
|
|
|
|
avcName,
|
|
|
|
)
|
|
|
|
|
2022-04-13 22:17:59 +02:00
|
|
|
default:
|
|
|
|
format := "format=yuv420p"
|
2022-04-06 17:46:41 +02:00
|
|
|
result = exec.Command(
|
|
|
|
ffmpegBin,
|
|
|
|
"-i", fileName,
|
|
|
|
"-c:v", string(encoder),
|
2023-03-02 03:30:45 +01:00
|
|
|
"-map", "0:v:0",
|
2022-04-06 17:46:41 +02:00
|
|
|
"-c:a", "aac",
|
|
|
|
"-vf", format,
|
|
|
|
"-max_muxing_queue_size", "1024",
|
|
|
|
"-crf", "23",
|
|
|
|
"-vsync", "vfr",
|
|
|
|
"-r", "30",
|
|
|
|
"-b:v", bitrate,
|
|
|
|
"-f", "mp4",
|
|
|
|
"-y",
|
|
|
|
avcName,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, useMutex, nil
|
|
|
|
}
|