Now handling audio samples in frame data as well

This commit is contained in:
Quentin Renard
2024-10-16 16:12:54 +02:00
parent 2a48086a32
commit c8b4cbec66
18 changed files with 381 additions and 218 deletions

View File

@@ -27,7 +27,7 @@ Examples are located in the [examples](examples) directory and mirror as much as
|Custom IO Demuxing|[see](examples/custom_io_demuxing/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/avio_reading.c) |Custom IO Demuxing|[see](examples/custom_io_demuxing/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/avio_reading.c)
|Demuxing/Decoding|[see](examples/demuxing_decoding/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/demuxing_decoding.c) |Demuxing/Decoding|[see](examples/demuxing_decoding/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/demuxing_decoding.c)
|Filtering|[see](examples/filtering/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/filtering_video.c) |Filtering|[see](examples/filtering/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/filtering_video.c)
|Frame data manipulating|[see](examples/frame_data_manipulating/main.go)|X |Frame data manipulation|[see](examples/frame_data_manipulation/main.go)|X
|Hardware Decoding|[see](examples/hardware_decoding/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/hw_decode.c) |Hardware Decoding|[see](examples/hardware_decoding/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/hw_decode.c)
|Remuxing|[see](examples/remuxing/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/remuxing.c) |Remuxing|[see](examples/remuxing/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/remuxing.c)
|Scaling|[see](examples/scaling/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/scaling_video.c) |Scaling|[see](examples/scaling/main.go)|[see](https://github.com/FFmpeg/FFmpeg/blob/n7.0/doc/examples/scaling_video.c)

View File

@@ -50,7 +50,7 @@ type helperInput struct {
lastFrame *Frame lastFrame *Frame
} }
func (h *helper) inputFormatContext(name string) (fc *FormatContext, err error) { func (h *helper) inputFormatContext(name string, ifmt *InputFormat) (fc *FormatContext, err error) {
h.m.Lock() h.m.Lock()
i, ok := h.inputs[name] i, ok := h.inputs[name]
if ok && i.formatContext != nil { if ok && i.formatContext != nil {
@@ -65,7 +65,7 @@ func (h *helper) inputFormatContext(name string) (fc *FormatContext, err error)
} }
h.closer.Add(fc.Free) h.closer.Add(fc.Free)
if err = fc.OpenInput("testdata/"+name, nil, nil); err != nil { if err = fc.OpenInput("testdata/"+name, ifmt, nil); err != nil {
err = fmt.Errorf("astiav_test: opening input failed: %w", err) err = fmt.Errorf("astiav_test: opening input failed: %w", err)
return return
} }
@@ -95,7 +95,7 @@ func (h *helper) inputFirstPacket(name string) (pkt *Packet, err error) {
h.m.Unlock() h.m.Unlock()
var fc *FormatContext var fc *FormatContext
if fc, err = h.inputFormatContext(name); err != nil { if fc, err = h.inputFormatContext(name, nil); err != nil {
err = fmt.Errorf("astiav_test: getting input format context failed") err = fmt.Errorf("astiav_test: getting input format context failed")
return return
} }
@@ -118,7 +118,7 @@ func (h *helper) inputFirstPacket(name string) (pkt *Packet, err error) {
return return
} }
func (h *helper) inputLastFrame(name string, mediaType MediaType) (f *Frame, err error) { func (h *helper) inputLastFrame(name string, mediaType MediaType, ifmt *InputFormat) (f *Frame, err error) {
h.m.Lock() h.m.Lock()
i, ok := h.inputs[name] i, ok := h.inputs[name]
if ok && i.lastFrame != nil { if ok && i.lastFrame != nil {
@@ -128,7 +128,7 @@ func (h *helper) inputLastFrame(name string, mediaType MediaType) (f *Frame, err
h.m.Unlock() h.m.Unlock()
var fc *FormatContext var fc *FormatContext
if fc, err = h.inputFormatContext(name); err != nil { if fc, err = h.inputFormatContext(name, ifmt); err != nil {
err = fmt.Errorf("astiav_test: getting input format context failed: %w", err) err = fmt.Errorf("astiav_test: getting input format context failed: %w", err)
return return
} }

View File

@@ -7,7 +7,7 @@ import (
) )
func TestCodecContext(t *testing.T) { func TestCodecContext(t *testing.T) {
fc, err := globalHelper.inputFormatContext("video.mp4") fc, err := globalHelper.inputFormatContext("video.mp4", nil)
require.NoError(t, err) require.NoError(t, err)
ss := fc.Streams() ss := fc.Streams()
require.Len(t, ss, 2) require.Len(t, ss, 2)

View File

@@ -7,7 +7,7 @@ import (
) )
func TestCodecParameters(t *testing.T) { func TestCodecParameters(t *testing.T) {
fc, err := globalHelper.inputFormatContext("video.mp4") fc, err := globalHelper.inputFormatContext("video.mp4", nil)
require.NoError(t, err) require.NoError(t, err)
ss := fc.Streams() ss := fc.Streams()
require.Len(t, ss, 2) require.Len(t, ss, 2)

View File

@@ -1,108 +0,0 @@
package main
import (
"fmt"
"image/png"
"log"
"os"
"strings"
"github.com/asticode/go-astiav"
)
const (
align = 1
pngPath = "testdata/image-rgba.png"
rawBufferPath = "testdata/image-rgba.rgba"
)
func main() {
// Handle ffmpeg logs
astiav.SetLogLevel(astiav.LogLevelDebug)
astiav.SetLogCallback(func(c astiav.Classer, l astiav.LogLevel, fmt, msg string) {
var cs string
if c != nil {
if cl := c.Class(); cl != nil {
cs = " - class: " + cl.String()
}
}
log.Printf("ffmpeg log: %s%s - level: %d\n", strings.TrimSpace(msg), cs, l)
})
// Alloc frames
f1 := astiav.AllocFrame()
defer f1.Free()
f2 := astiav.AllocFrame()
defer f2.Free()
// To write data manually into a frame, proper attributes need to be set and allocated
for _, f := range []*astiav.Frame{f1, f2} {
// Set attributes
f.SetHeight(256)
f.SetPixelFormat(astiav.PixelFormatRgba)
f.SetWidth(256)
// Alloc buffer
if err := f.AllocBuffer(align); err != nil {
log.Fatal(fmt.Errorf("main: allocating buffer failed: %w", err))
}
// Alloc image
if err := f.AllocImage(align); err != nil {
log.Fatal(fmt.Errorf("main: allocating image failed: %w", err))
}
}
// When writing data manually into a frame, you usually need to make sure the frame is writable
// Don't forget this step above all if the frame's buffer is referenced elsewhere
for _, f := range []*astiav.Frame{f1, f2} {
// Make writable
if err := f.MakeWritable(); err != nil {
log.Fatal(fmt.Errorf("main: making frame writable failed: %w", err))
}
}
// As an example, we're going to write data manually into the first frame based on a buffer (i.e. raw data)
b, err := os.ReadFile(rawBufferPath)
if err != nil {
log.Fatal(fmt.Errorf("main: reading %s failed: %w", rawBufferPath, err))
}
if err := f1.Data().SetBytes(b, align); err != nil {
log.Fatal(fmt.Errorf("main: setting frame's data based on bytes failed: %w", err))
}
// As an example, we're going to write data manually into the second frame based on a Go image
fl1, err := os.Open(pngPath)
if err != nil {
log.Fatal(fmt.Errorf("main: opening %s failed: %w", pngPath, err))
}
defer fl1.Close()
i1, err := png.Decode(fl1)
if err != nil {
log.Fatal(fmt.Errorf("main: decoding %s failed: %w", pngPath, err))
}
if err := f2.Data().FromImage(i1); err != nil {
log.Fatal(fmt.Errorf("main: setting frame's data based on Go image failed: %w", err))
}
// This is the place where you do stuff with the frames
// As an example, we're going to read the first frame's data as a buffer (i.e. raw data)
if _, err = f1.Data().Bytes(align); err != nil {
log.Fatal(fmt.Errorf("main: getting frame's data as bytes failed: %w", err))
}
// As an example, we're going to read the second frame's data as a Go image
// For that we first need to guess the Go image format based on the frame's attributes before providing
// it to .ToImage(). You may not need this and can provide your own image.Image to .ToImage()
i2, err := f2.Data().GuessImageFormat()
if err != nil {
log.Fatal(fmt.Errorf("main: guessing image format failed: %w", err))
}
if err := f2.Data().ToImage(i2); err != nil {
log.Fatal(fmt.Errorf("main: getting frame's data as Go image failed: %w", err))
}
// Success
log.Println("success")
}

View File

@@ -0,0 +1,129 @@
package main
import (
"fmt"
"image"
"log"
"strings"
"github.com/asticode/go-astiav"
)
func main() {
// Handle ffmpeg logs
astiav.SetLogLevel(astiav.LogLevelDebug)
astiav.SetLogCallback(func(c astiav.Classer, l astiav.LogLevel, fmt, msg string) {
var cs string
if c != nil {
if cl := c.Class(); cl != nil {
cs = " - class: " + cl.String()
}
}
log.Printf("ffmpeg log: %s%s - level: %d\n", strings.TrimSpace(msg), cs, l)
})
/*
In this first part we're going to manipulate an audio frame
*/
// Alloc frame
audioFrame := astiav.AllocFrame()
defer audioFrame.Free()
// To write data manually into a frame, proper attributes need to be set and allocated
audioFrame.SetChannelLayout(astiav.ChannelLayoutStereo)
audioFrame.SetNbSamples(960)
audioFrame.SetSampleFormat(astiav.SampleFormatFlt)
audioFrame.SetSampleRate(48000)
// Alloc buffer
align := 0
if err := audioFrame.AllocBuffer(align); err != nil {
log.Fatal(fmt.Errorf("main: allocating buffer failed: %w", err))
}
// Alloc samples
if err := audioFrame.AllocSamples(align); err != nil {
log.Fatal(fmt.Errorf("main: allocating image failed: %w", err))
}
// When writing data manually into a frame, you need to make sure the frame is writable
if err := audioFrame.MakeWritable(); err != nil {
log.Fatal(fmt.Errorf("main: making frame writable failed: %w", err))
}
// Let's say b1 contains an actual audio buffer, we can update the audio frame's data based on the buffer
var b1 []byte
if err := audioFrame.Data().SetBytes(b1, align); err != nil {
log.Fatal(fmt.Errorf("main: setting frame's data based from bytes failed: %w", err))
}
// We can also retrieve the audio frame's data as buffer
if _, err := audioFrame.Data().Bytes(align); err != nil {
log.Fatal(fmt.Errorf("main: getting frame's data as bytes failed: %w", err))
}
/*
In this second part we're going to manipulate a video frame
*/
// Alloc frame
videoFrame := astiav.AllocFrame()
defer videoFrame.Free()
// To write data manually into a frame, proper attributes need to be set and allocated
videoFrame.SetHeight(256)
videoFrame.SetPixelFormat(astiav.PixelFormatRgba)
videoFrame.SetWidth(256)
// Alloc buffer
align = 1
if err := videoFrame.AllocBuffer(align); err != nil {
log.Fatal(fmt.Errorf("main: allocating buffer failed: %w", err))
}
// Alloc image
if err := videoFrame.AllocImage(align); err != nil {
log.Fatal(fmt.Errorf("main: allocating image failed: %w", err))
}
// When writing data manually into a frame, you need to make sure the frame is writable
if err := videoFrame.MakeWritable(); err != nil {
log.Fatal(fmt.Errorf("main: making frame writable failed: %w", err))
}
// Let's say b2 contains an actual video buffer, we can update the video frame's data based on the buffer
var b2 []byte
if err := videoFrame.Data().SetBytes(b2, align); err != nil {
log.Fatal(fmt.Errorf("main: setting frame's data based from bytes failed: %w", err))
}
// We can also retrieve the video frame's data as buffer
if _, err := videoFrame.Data().Bytes(align); err != nil {
log.Fatal(fmt.Errorf("main: getting frame's data as bytes failed: %w", err))
}
// Let's say i1 is an actual Go image.Image, we can update the video frame's data based on the image
var i1 image.Image
if err := videoFrame.Data().FromImage(i1); err != nil {
log.Fatal(fmt.Errorf("main: setting frame's data based on Go image failed: %w", err))
}
// We can also retrieve the video frame's data as a Go image
// For that we first need to guess the Go image format based on the frame's attributes before providing
// it to .ToImage(). You may not need this and can provide your own image.Image to .ToImage()
i2, err := videoFrame.Data().GuessImageFormat()
if err != nil {
log.Fatal(fmt.Errorf("main: guessing image format failed: %w", err))
}
if err := videoFrame.Data().ToImage(i2); err != nil {
log.Fatal(fmt.Errorf("main: getting frame's data as Go image failed: %w", err))
}
// Success
log.Println("success")
}

View File

@@ -7,7 +7,7 @@ import (
) )
func TestFormatContext(t *testing.T) { func TestFormatContext(t *testing.T) {
fc1, err := globalHelper.inputFormatContext("video.mp4") fc1, err := globalHelper.inputFormatContext("video.mp4", nil)
require.NoError(t, err) require.NoError(t, err)
ss := fc1.Streams() ss := fc1.Streams()
require.Len(t, ss, 2) require.Len(t, ss, 2)

22
frame.c Normal file
View File

@@ -0,0 +1,22 @@
#include <errno.h>
#include <libavutil/avutil.h>
#include <libavutil/samplefmt.h>
#include <stdint.h>
#include <string.h>
int astiavSamplesCopyToBuffer(uint8_t* dst, int dst_size, const uint8_t * const src_data[8], int nb_channels, int nb_samples, enum AVSampleFormat sample_fmt, int align) {
int linesize, buffer_size, nb_planes, i;
buffer_size = av_samples_get_buffer_size(&linesize, nb_channels, nb_samples, sample_fmt, align);
if (buffer_size > dst_size || buffer_size < 0) return AVERROR(EINVAL);
nb_planes = buffer_size / linesize;
for (i = 0; i < nb_planes; i++) {
const uint8_t *src = src_data[i];
memcpy(dst, src, linesize);
dst += linesize;
src += linesize;
}
return buffer_size;
}

View File

@@ -5,6 +5,7 @@ package astiav
//#include <libavutil/imgutils.h> //#include <libavutil/imgutils.h>
//#include <libavutil/samplefmt.h> //#include <libavutil/samplefmt.h>
//#include <libavutil/hwcontext.h> //#include <libavutil/hwcontext.h>
//#include "frame.h"
import "C" import "C"
import ( import (
"unsafe" "unsafe"
@@ -113,6 +114,26 @@ func (f *Frame) ImageFillBlack() error {
return newError(C.av_image_fill_black(&f.c.data[0], &linesize[0], (C.enum_AVPixelFormat)(f.c.format), (C.enum_AVColorRange)(f.c.color_range), f.c.width, f.c.height)) return newError(C.av_image_fill_black(&f.c.data[0], &linesize[0], (C.enum_AVPixelFormat)(f.c.format), (C.enum_AVColorRange)(f.c.color_range), f.c.width, f.c.height))
} }
func (f *Frame) SamplesBufferSize(align int) (int, error) {
ret := C.av_samples_get_buffer_size(nil, f.c.ch_layout.nb_channels, f.c.nb_samples, (C.enum_AVSampleFormat)(f.c.format), C.int(align))
if err := newError(ret); err != nil {
return 0, err
}
return int(ret), nil
}
func (f *Frame) SamplesCopyToBuffer(b []byte, align int) (int, error) {
ret := C.astiavSamplesCopyToBuffer((*C.uint8_t)(unsafe.Pointer(&b[0])), C.int(len(b)), &f.c.data[0], f.c.ch_layout.nb_channels, f.c.nb_samples, (C.enum_AVSampleFormat)(f.c.format), C.int(align))
if err := newError(ret); err != nil {
return 0, err
}
return int(ret), nil
}
func (f *Frame) SamplesFillSilence() error {
return newError(C.av_samples_set_silence(&f.c.data[0], 0, f.c.nb_samples, f.c.ch_layout.nb_channels, (C.enum_AVSampleFormat)(f.c.format)))
}
func (f *Frame) Linesize() [NumDataPointers]int { func (f *Frame) Linesize() [NumDataPointers]int {
o := [NumDataPointers]int{} o := [NumDataPointers]int{}
for i := 0; i < int(NumDataPointers); i++ { for i := 0; i < int(NumDataPointers); i++ {

4
frame.h Normal file
View File

@@ -0,0 +1,4 @@
#include <libavutil/samplefmt.h>
#include <stdint.h>
int astiavSamplesCopyToBuffer(uint8_t* dst, int dst_size, const uint8_t * const src_data[8], int nb_channels, int nb_samples, enum AVSampleFormat sample_fmt, int align);

View File

@@ -1,6 +1,7 @@
package astiav package astiav
//#include <libavutil/imgutils.h> //#include <libavutil/imgutils.h>
//#include <libavutil/samplefmt.h>
//#include <stdlib.h> //#include <stdlib.h>
//#include "macros.h" //#include "macros.h"
import "C" import "C"
@@ -254,30 +255,39 @@ func newFrameDataFrame(f *Frame) *frameDataFrame {
} }
func (f *frameDataFrame) bytes(align int) ([]byte, error) { func (f *frameDataFrame) bytes(align int) ([]byte, error) {
switch { // Get funcs
// Video var bufferSizeFunc func(int) (int, error)
case f.height() > 0 && f.width() > 0: var copyToBufferFunc func([]byte, int) (int, error)
// Get buffer size switch f.mediaType() {
s, err := f.f.ImageBufferSize(align) case MediaTypeAudio:
if err != nil { bufferSizeFunc = f.f.SamplesBufferSize
return nil, fmt.Errorf("astiav: getting image buffer size failed: %w", err) copyToBufferFunc = f.f.SamplesCopyToBuffer
} case MediaTypeVideo:
bufferSizeFunc = f.f.ImageBufferSize
// Invalid buffer size copyToBufferFunc = f.f.ImageCopyToBuffer
if s == 0 { default:
return nil, errors.New("astiav: invalid image buffer size") return nil, errors.New("astiav: media type not implemented")
}
// Create buffer
b := make([]byte, s)
// Copy image to buffer
if _, err = f.f.ImageCopyToBuffer(b, align); err != nil {
return nil, fmt.Errorf("astiav: copying image to buffer failed: %w", err)
}
return b, nil
} }
return nil, errors.New("astiav: frame type not implemented")
// Get buffer size
s, err := bufferSizeFunc(align)
if err != nil {
return nil, fmt.Errorf("astiav: getting buffer size failed: %w", err)
}
// Invalid buffer size
if s == 0 {
return nil, errors.New("astiav: invalid buffer size")
}
// Create buffer
b := make([]byte, s)
// Copy to buffer
if _, err = copyToBufferFunc(b, align); err != nil {
return nil, fmt.Errorf("astiav: copying to buffer failed: %w", err)
}
return b, nil
} }
func (f *frameDataFrame) copyPlanes(ps []frameDataPlane) error { func (f *frameDataFrame) copyPlanes(ps []frameDataPlane) error {
@@ -286,26 +296,28 @@ func (f *frameDataFrame) copyPlanes(ps []frameDataPlane) error {
return errors.New("astiav: frame is not writable") return errors.New("astiav: frame is not writable")
} }
switch { // Prepare data
// Video var cdata [8]*C.uint8_t
case f.height() > 0 && f.width() > 0: var clinesizes [8]C.int
// Loop through planes for i, p := range ps {
var cdata [8]*C.uint8_t // Convert data
var clinesizes [8]C.int if len(p.bytes) > 0 {
for i, p := range ps { cdata[i] = (*C.uint8_t)(C.CBytes(p.bytes))
// Convert data defer C.free(unsafe.Pointer(cdata[i]))
if len(p.bytes) > 0 {
cdata[i] = (*C.uint8_t)(C.CBytes(p.bytes))
defer C.free(unsafe.Pointer(cdata[i]))
}
// Convert linesize
clinesizes[i] = C.int(p.linesize)
} }
// Copy image // Convert linesize
clinesizes[i] = C.int(p.linesize)
}
// Copy data
switch f.mediaType() {
case MediaTypeAudio:
C.av_samples_copy(&f.f.c.data[0], &cdata[0], 0, 0, f.f.c.nb_samples, f.f.c.ch_layout.nb_channels, (C.enum_AVSampleFormat)(f.f.c.format))
case MediaTypeVideo:
C.av_image_copy(&f.f.c.data[0], &f.f.c.linesize[0], &cdata[0], &clinesizes[0], (C.enum_AVPixelFormat)(f.f.c.format), f.f.c.width, f.f.c.height) C.av_image_copy(&f.f.c.data[0], &f.f.c.linesize[0], &cdata[0], &clinesizes[0], (C.enum_AVPixelFormat)(f.f.c.format), f.f.c.width, f.f.c.height)
return nil default:
return errors.New("astiav: media type not implemented")
} }
return nil return nil
} }
@@ -314,57 +326,91 @@ func (f *frameDataFrame) height() int {
return f.f.Height() return f.f.Height()
} }
func (f *frameDataFrame) mediaType() MediaType {
switch {
// Audio
case f.f.NbSamples() > 0:
return MediaTypeAudio
// Video
case f.f.Height() > 0 && f.f.Width() > 0:
return MediaTypeVideo
default:
return MediaTypeUnknown
}
}
func (f *frameDataFrame) pixelFormat() PixelFormat { func (f *frameDataFrame) pixelFormat() PixelFormat {
return f.f.PixelFormat() return f.f.PixelFormat()
} }
func (f *frameDataFrame) planes(b []byte, align int) ([]frameDataPlane, error) { func (f *frameDataFrame) planes(b []byte, align int) ([]frameDataPlane, error) {
switch { // Get line and plane sizes
// Video var linesizes [8]int
case f.height() > 0 && f.width() > 0: var planeSizes [8]int
switch f.mediaType() {
case MediaTypeAudio:
// Get buffer size
var cLinesize C.int
cBufferSize := C.av_samples_get_buffer_size(&cLinesize, f.f.c.ch_layout.nb_channels, f.f.c.nb_samples, (C.enum_AVSampleFormat)(f.f.c.format), C.int(align))
if err := newError(cBufferSize); err != nil {
return nil, fmt.Errorf("astiav: getting buffer size failed: %w", err)
}
// Update line and plane sizes
for i := 0; i < int(cBufferSize/cLinesize); i++ {
linesizes[i] = int(cLinesize)
planeSizes[i] = int(cLinesize)
}
case MediaTypeVideo:
// Below is mostly inspired by https://github.com/FFmpeg/FFmpeg/blob/n5.1.2/libavutil/imgutils.c#L466 // Below is mostly inspired by https://github.com/FFmpeg/FFmpeg/blob/n5.1.2/libavutil/imgutils.c#L466
// Get linesize // Get linesize
var linesizes [4]C.int var cLinesizes [8]C.int
if err := newError(C.av_image_fill_linesizes(&linesizes[0], (C.enum_AVPixelFormat)(f.f.c.format), f.f.c.width)); err != nil { if err := newError(C.av_image_fill_linesizes(&cLinesizes[0], (C.enum_AVPixelFormat)(f.f.c.format), f.f.c.width)); err != nil {
return nil, fmt.Errorf("astiav: getting linesize failed: %w", err) return nil, fmt.Errorf("astiav: getting linesize failed: %w", err)
} }
// Align linesize // Align linesize
var alignedLinesizes [4]C.ptrdiff_t var cAlignedLinesizes [8]C.ptrdiff_t
for i := 0; i < 4; i++ { for i := 0; i < 4; i++ {
alignedLinesizes[i] = C.astiavFFAlign(linesizes[i], C.int(align)) cAlignedLinesizes[i] = C.astiavFFAlign(cLinesizes[i], C.int(align))
} }
// Get plane sizes // Get plane sizes
var planeSizes [4]C.size_t var cPlaneSizes [8]C.size_t
if err := newError(C.av_image_fill_plane_sizes(&planeSizes[0], (C.enum_AVPixelFormat)(f.f.c.format), f.f.c.height, &alignedLinesizes[0])); err != nil { if err := newError(C.av_image_fill_plane_sizes(&cPlaneSizes[0], (C.enum_AVPixelFormat)(f.f.c.format), f.f.c.height, &cAlignedLinesizes[0])); err != nil {
return nil, fmt.Errorf("astiav: getting plane sizes failed: %w", err) return nil, fmt.Errorf("astiav: getting plane sizes failed: %w", err)
} }
// Loop through planes // Update line and plane sizes
var ps []frameDataPlane for i := range cPlaneSizes {
start := 0 linesizes[i] = int(cAlignedLinesizes[i])
for i := range planeSizes { planeSizes[i] = int(cPlaneSizes[i])
// Get end
end := start + int(planeSizes[i])
if len(b) < end {
return nil, fmt.Errorf("astiav: buffer length %d is invalid for [%d:%d]", len(b), start, end)
}
// Append plane
ps = append(ps, frameDataPlane{
bytes: b[start:end],
linesize: int(linesizes[i]),
})
// Update start
start = end
} }
return ps, nil
default: default:
return nil, errors.New("astiav: frame type not implemented") return nil, errors.New("astiav: media type not implemented")
} }
// Loop through plane sizes
var ps []frameDataPlane
start := 0
for i := range planeSizes {
// Get end
end := start + planeSizes[i]
if len(b) < end {
return nil, fmt.Errorf("astiav: buffer length %d is invalid for [%d:%d]", len(b), start, end)
}
// Append plane
ps = append(ps, frameDataPlane{
bytes: b[start:end],
linesize: linesizes[i],
})
// Update start
start = end
}
return ps, nil
} }
func (f *frameDataFrame) width() int { func (f *frameDataFrame) width() int {

View File

@@ -119,14 +119,14 @@ func TestFrameDataInternal(t *testing.T) {
} }
} }
fdf.h = 1
b1 := []byte{0, 1, 2, 3} b1 := []byte{0, 1, 2, 3}
fdf.onBytes = func(align int) ([]byte, error) { return b1, nil } fdf.onBytes = func(align int) ([]byte, error) { return b1, nil }
fdf.w = 2
b2, err := fd.Bytes(0) b2, err := fd.Bytes(0)
require.NoError(t, err) require.NoError(t, err)
require.Equal(t, b1, b2) require.Equal(t, b1, b2)
fdf.h = 1
fdf.w = 2
for _, v := range []struct { for _, v := range []struct {
e image.Image e image.Image
i image.Image i image.Image
@@ -465,18 +465,28 @@ func TestFrameDataInternal(t *testing.T) {
func TestFrameData(t *testing.T) { func TestFrameData(t *testing.T) {
for _, v := range []struct { for _, v := range []struct {
ext string ext string
ifmt *InputFormat
md MediaType
name string name string
}{ }{
{
ext: "pcm",
ifmt: FindInputFormat("s16le"),
md: MediaTypeAudio,
name: "audio-s16le",
},
{ {
ext: "png", ext: "png",
md: MediaTypeVideo,
name: "image-rgba", name: "image-rgba",
}, },
{ {
ext: "h264", ext: "h264",
md: MediaTypeVideo,
name: "video-yuv420p", name: "video-yuv420p",
}, },
} { } {
f1, err := globalHelper.inputLastFrame(v.name+"."+v.ext, MediaTypeVideo) f1, err := globalHelper.inputLastFrame(v.name+"."+v.ext, v.md, v.ifmt)
require.NoError(t, err) require.NoError(t, err)
fd1 := f1.Data() fd1 := f1.Data()
@@ -487,31 +497,55 @@ func TestFrameData(t *testing.T) {
require.NoError(t, err) require.NoError(t, err)
require.Equal(t, b3, b2) require.Equal(t, b3, b2)
i1, err := fd1.GuessImageFormat() var i1 image.Image
require.NoError(t, err) switch v.md {
require.NoError(t, fd1.ToImage(i1)) case MediaTypeVideo:
b4 := []byte(fmt.Sprintf("%+v", i1)) i1, err = fd1.GuessImageFormat()
b5, err := os.ReadFile("testdata/" + v.name + "-struct") require.NoError(t, err)
require.NoError(t, err) require.NoError(t, fd1.ToImage(i1))
require.Equal(t, b5, b4) b4 := []byte(fmt.Sprintf("%+v", i1))
b5, err := os.ReadFile("testdata/" + v.name + "-struct")
require.NoError(t, err)
require.Equal(t, b5, b4)
}
f2 := AllocFrame() f2 := AllocFrame()
defer f2.Free() defer f2.Free()
f2.SetHeight(f1.Height())
f2.SetPixelFormat(f1.PixelFormat())
f2.SetWidth(f1.Width())
const align = 1
require.NoError(t, f2.AllocBuffer(align))
require.NoError(t, f2.AllocImage(align))
fd2 := f2.Data() fd2 := f2.Data()
require.NoError(t, fd2.FromImage(i1)) align := 0
b6, err := fd2.Bytes(align) switch v.md {
require.NoError(t, err) case MediaTypeAudio:
b7 := []byte(fmt.Sprintf("%+v", b6)) f2.SetChannelLayout(f1.ChannelLayout())
require.Equal(t, b3, b7) f2.SetNbSamples(f1.NbSamples())
f2.SetSampleFormat(f1.SampleFormat())
f2.SetSampleRate(f1.SampleRate())
require.NoError(t, f2.AllocBuffer(align))
require.NoError(t, f2.AllocSamples(align))
case MediaTypeVideo:
align = 1
f2.SetHeight(f1.Height())
f2.SetPixelFormat(f1.PixelFormat())
f2.SetWidth(f1.Width())
require.NoError(t, f2.AllocBuffer(align))
require.NoError(t, f2.AllocImage(align))
}
require.NoError(t, f2.ImageFillBlack()) switch v.md {
case MediaTypeVideo:
require.NoError(t, fd2.FromImage(i1))
b6, err := fd2.Bytes(align)
require.NoError(t, err)
b7 := []byte(fmt.Sprintf("%+v", b6))
require.Equal(t, b3, b7)
}
switch v.md {
case MediaTypeAudio:
require.NoError(t, f2.SamplesFillSilence())
case MediaTypeVideo:
require.NoError(t, f2.ImageFillBlack())
}
require.NoError(t, fd2.SetBytes(b1, align)) require.NoError(t, fd2.SetBytes(b1, align))
b1[0] -= 1 b1[0] -= 1
b8, err := fd2.Bytes(align) b8, err := fd2.Bytes(align)
@@ -522,10 +556,16 @@ func TestFrameData(t *testing.T) {
f3 := AllocFrame() f3 := AllocFrame()
defer f3.Free() defer f3.Free()
require.NoError(t, f3.Ref(f2)) require.NoError(t, f3.Ref(f2))
require.Error(t, fd2.FromImage(i1)) switch v.md {
case MediaTypeVideo:
require.Error(t, fd2.FromImage(i1))
}
require.Error(t, fd2.SetBytes(b1, align)) require.Error(t, fd2.SetBytes(b1, align))
f2.MakeWritable() f2.MakeWritable()
require.NoError(t, fd2.FromImage(i1)) switch v.md {
case MediaTypeVideo:
require.NoError(t, fd2.FromImage(i1))
}
require.NoError(t, fd2.SetBytes(b1, align)) require.NoError(t, fd2.SetBytes(b1, align))
} }
} }

File diff suppressed because one or more lines are too long

View File

@@ -93,7 +93,7 @@ func TestSoftwareScaleContext(t *testing.T) {
require.Equal(t, w, srcW) require.Equal(t, w, srcW)
require.Equal(t, h, srcH) require.Equal(t, h, srcH)
f4, err := globalHelper.inputLastFrame("image-rgba.png", MediaTypeVideo) f4, err := globalHelper.inputLastFrame("image-rgba.png", MediaTypeVideo, nil)
require.NoError(t, err) require.NoError(t, err)
f5 := AllocFrame() f5 := AllocFrame()

View File

@@ -7,7 +7,7 @@ import (
) )
func TestStream(t *testing.T) { func TestStream(t *testing.T) {
fc, err := globalHelper.inputFormatContext("video.mp4") fc, err := globalHelper.inputFormatContext("video.mp4", nil)
require.NoError(t, err) require.NoError(t, err)
ss := fc.Streams() ss := fc.Streams()
require.Len(t, ss, 2) require.Len(t, ss, 2)

1
testdata/audio-s16le-bytes vendored Executable file

File diff suppressed because one or more lines are too long

BIN
testdata/audio-s16le.pcm vendored Normal file

Binary file not shown.

Binary file not shown.