Skip to content

Commit

Permalink
Adds additional methods and cleans up export
Browse files Browse the repository at this point in the history
  • Loading branch information
davidbyttow committed Jan 6, 2021
1 parent cabe3e6 commit b2f6b21
Show file tree
Hide file tree
Showing 6 changed files with 274 additions and 279 deletions.
165 changes: 54 additions & 111 deletions vips/foreign.c
Original file line number Diff line number Diff line change
@@ -1,106 +1,92 @@
#include "lang.h"
#include "foreign.h"

int load_image_buffer(void *buf, size_t len, int imageType, VipsImage **out)
{
int load_image_buffer(void *buf, size_t len, int imageType, VipsImage **out) {
int code = 1;

if (imageType == JPEG)
{
if (imageType == JPEG) {
code = vips_jpegload_buffer(buf, len, out, NULL);
}
else if (imageType == PNG)
{
} else if (imageType == PNG) {
code = vips_pngload_buffer(buf, len, out, NULL);
}
else if (imageType == WEBP)
{
} else if (imageType == WEBP) {
code = vips_webpload_buffer(buf, len, out, NULL);
}
else if (imageType == TIFF)
{
} else if (imageType == TIFF) {
code = vips_tiffload_buffer(buf, len, out, NULL);
}
else if (imageType == GIF)
{
} else if (imageType == GIF) {
code = vips_gifload_buffer(buf, len, out, NULL);
}
else if (imageType == PDF)
{
} else if (imageType == PDF) {
code = vips_pdfload_buffer(buf, len, out, NULL);
}
else if (imageType == SVG)
{
} else if (imageType == SVG) {
code = vips_svgload_buffer(buf, len, out, NULL);
}
else if (imageType == HEIF)
{
} else if (imageType == HEIF) {
// added autorotate on load as currently it addresses orientation issues
// https://github.com/libvips/libvips/pull/1680
code = vips_heifload_buffer(buf, len, out, "autorotate", TRUE, NULL);
}
else if (imageType == MAGICK)
{
} else if (imageType == MAGICK) {
code = vips_magickload_buffer(buf, len, out, NULL);
}

return code;
}

// https://libvips.github.io/libvips/API/current/VipsForeignSave.html#vips-jpegsave-buffer
int save_jpeg_buffer(VipsImage *in, void **buf, size_t *len, int strip, int quality, int interlace)
{
return vips_jpegsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"Q", quality,
int save_jpeg_buffer(SaveParams *params) {
return vips_jpegsave_buffer(params->inputImage,
&params->outputBuffer,
&params->outputLen,
"strip", INT_TO_GBOOLEAN(params->stripMetadata),
"Q", params->quality,
"optimize_coding", TRUE,
"interlace", INT_TO_GBOOLEAN(interlace),
"interlace", INT_TO_GBOOLEAN(params->interlace),
"subsample_mode", VIPS_FOREIGN_JPEG_SUBSAMPLE_ON,
NULL);
}

// https://libvips.github.io/libvips/API/current/VipsForeignSave.html#vips-pngsave-buffer
int save_png_buffer(VipsImage *in, void **buf, size_t *len, int strip, int compression, int interlace)
{
return vips_pngsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"compression", compression,
"interlace", INT_TO_GBOOLEAN(interlace),
int save_png_buffer(SaveParams *params) {
return vips_pngsave_buffer(params->inputImage,
&params->outputBuffer,
&params->outputLen,
"strip", INT_TO_GBOOLEAN(params->stripMetadata),
"compression", params->pngCompression,
"interlace", INT_TO_GBOOLEAN(params->interlace),
"filter", VIPS_FOREIGN_PNG_FILTER_NONE,
NULL);
}

// todo: support additional params
// https://github.com/libvips/libvips/blob/master/libvips/foreign/webpsave.c#L524
// https://libvips.github.io/libvips/API/current/VipsForeignSave.html#vips-webpsave-buffer
int save_webp_buffer(VipsImage *in, void **buf, size_t *len, int strip, int quality, int lossless, int effort)
{
return vips_webpsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"Q", quality,
"lossless", INT_TO_GBOOLEAN(lossless),
"reduction_effort", effort,
int save_webp_buffer(SaveParams *params) {
return vips_webpsave_buffer(params->inputImage,
&params->outputBuffer,
&params->outputLen,
"strip", INT_TO_GBOOLEAN(params->stripMetadata),
"Q", params->quality,
"lossless", INT_TO_GBOOLEAN(params->webpLossless),
"reduction_effort", params->webpReductionEffort,
NULL);
}

// todo: support additional params
// https://github.com/libvips/libvips/blob/master/libvips/foreign/heifsave.c#L653
int save_heif_buffer(VipsImage *in, void **buf, size_t *len, int quality, int lossless)
{
return vips_heifsave_buffer(in, buf, len,
"Q", quality,
"lossless", INT_TO_GBOOLEAN(lossless),
int save_heif_buffer(SaveParams *params) {
return vips_heifsave_buffer(params->inputImage,
&params->outputBuffer,
&params->outputLen,
"Q", params->quality,
"lossless", INT_TO_GBOOLEAN(params->heifLossless),
NULL);
}

// https://libvips.github.io/libvips/API/current/VipsForeignSave.html#vips-tiffsave-buffer
int save_tiff_buffer(VipsImage *in, void **buf, size_t *len, int strip, int quality, int lossless)
{
// TODO: Allow various options to be passed in.
return vips_tiffsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(strip),
"Q", quality,
"compression", lossless ? VIPS_FOREIGN_TIFF_COMPRESSION_NONE : VIPS_FOREIGN_TIFF_COMPRESSION_LZW,
int save_tiff_buffer(SaveParams *params) {
return vips_tiffsave_buffer(params->inputImage,
&params->outputBuffer,
&params->outputLen,
"strip", INT_TO_GBOOLEAN(params->stripMetadata),
"Q", params->quality,
"compression", params->tiffCompression,
"pyramid", FALSE,
"predictor", VIPS_FOREIGN_TIFF_PREDICTOR_HORIZONTAL,
"pyramid", FALSE,
Expand All @@ -112,65 +98,22 @@ int save_tiff_buffer(VipsImage *in, void **buf, size_t *len, int strip, int qual
NULL);
}

int save_to_buffer(SaveAsParams *params)
{
return 0;
int ret;
VipsImage *in = params->inputImage;
void **buf = params->outputBuffer;
size_t *len = params->outputLen;

int save_to_buffer(SaveParams *params) {
switch (params->outputFormat)
{
case JPEG:
ret = vips_jpegsave_buffer(params->inputImage, params->outputBuffer, params->outputLen,
"strip", INT_TO_GBOOLEAN(params->stripMetadata),
"Q", params->quality,
"optimize_coding", TRUE,
"interlace", INT_TO_GBOOLEAN(params->interlace),
"subsample_mode", VIPS_FOREIGN_JPEG_SUBSAMPLE_ON,
NULL);
break;
return save_jpeg_buffer(params);
case PNG:
ret = vips_pngsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(params->stripMetadata),
"compression", params->pngCompression,
"interlace", INT_TO_GBOOLEAN(params->interlace),
"filter", VIPS_FOREIGN_PNG_FILTER_NONE,
NULL);
break;
return save_png_buffer(params);
case WEBP:
ret = vips_webpsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(params->stripMetadata),
"Q", params->quality,
"lossless", INT_TO_GBOOLEAN(params->webpLossless),
"reduction_effort", params->webpReductionEffort,
NULL);
break;
return save_webp_buffer(params);
case HEIF:
ret = vips_heifsave_buffer(in, buf, len,
"Q", params->quality,
"lossless", INT_TO_GBOOLEAN(params->heifLossless),
NULL);
break;
return save_heif_buffer(params);
case TIFF:
ret = vips_tiffsave_buffer(in, buf, len,
"strip", INT_TO_GBOOLEAN(params->stripMetadata),
"Q", params->quality,
"compression", params->tiffCompression,
"pyramid", FALSE,
"predictor", VIPS_FOREIGN_TIFF_PREDICTOR_HORIZONTAL,
"pyramid", FALSE,
"tile", FALSE,
"tile_height", 256,
"tile_width", 256,
"xres", 1.0,
"yres", 1.0,
NULL);
break;
return save_tiff_buffer(params);
default:
ret = -1;
g_warning("Unsupported output type given: %d", params->outputFormat);
return -1;
}
}

return ret;
}
115 changes: 47 additions & 68 deletions vips/foreign.go
Original file line number Diff line number Diff line change
Expand Up @@ -236,104 +236,83 @@ func bmpToPNG(src []byte) ([]byte, error) {
return w.Bytes(), nil
}

func vipsSavePNGToBuffer(in *C.VipsImage, stripMetadata bool, compression int, interlaced bool) ([]byte, error) {
func vipsSavePNGToBuffer(in *C.VipsImage, params PngExportParams) ([]byte, error) {
incOpCounter("save_png_buffer")
var buf unsafe.Pointer
len := C.size_t(0)

// strip := C.int(boolToInt(stripMetadata))
// comp := C.int(compression)
// inter := C.int(boolToInt(interlaced))

params := C.struct_SaveAsParams{
p := C.struct_SaveParams{
inputImage: in,
outputBuffer: &buf,
outputLen: &len,
stripMetadata: C.int(boolToInt(stripMetadata)),
interlace: C.int(boolToInt(interlaced)),
pngCompression: C.int(compression),
}

var ptr = (*C.struct_SaveAsParams)(unsafe.Pointer(&params))

// if err := C.save_png_buffer(in, &ptr, &cLen, strip, comp, inter); err != 0 {
// return nil, handleSaveBufferError(ptr)
// }

if err := C.save_to_buffer(ptr); err != 0 {
return nil, handleSaveBufferError(buf)
outputFormat: C.PNG,
stripMetadata: C.int(boolToInt(params.StripMetadata)),
interlace: C.int(boolToInt(params.Interlace)),
pngCompression: C.int(params.Compression),
}

return toBuff(buf, len), nil
return vipsSaveToBuffer(p)
}

func vipsSaveWebPToBuffer(in *C.VipsImage, stripMetadata bool, quality int, lossless bool, effort int) ([]byte, error) {
func vipsSaveWebPToBuffer(in *C.VipsImage, params WebpExportParams) ([]byte, error) {
incOpCounter("save_webp_buffer")
var ptr unsafe.Pointer
cLen := C.size_t(0)

strip := C.int(boolToInt(stripMetadata))
qual := C.int(quality)
loss := C.int(boolToInt(lossless))
eff := C.int(effort)

if err := C.save_webp_buffer(in, &ptr, &cLen, strip, qual, loss, eff); err != 0 {
return nil, handleSaveBufferError(ptr)
p := C.struct_SaveParams{
inputImage: in,
outputFormat: C.WEBP,
stripMetadata: C.int(boolToInt(params.StripMetadata)),
quality: C.int(params.Quality),
webpLossless: C.int(boolToInt(params.Lossless)),
webpReductionEffort: C.int(params.ReductionEffort),
}

return toBuff(ptr, cLen), nil
return vipsSaveToBuffer(p)
}

func vipsSaveTIFFToBuffer(in *C.VipsImage, stripMetadata bool, quality int, lossless bool) ([]byte, error) {
func vipsSaveTIFFToBuffer(in *C.VipsImage, params TiffExportParams) ([]byte, error) {
incOpCounter("save_tiff_buffer")
var ptr unsafe.Pointer
cLen := C.size_t(0)

strip := C.int(boolToInt(stripMetadata))
qual := C.int(quality)
loss := C.int(boolToInt(lossless))

if err := C.save_tiff_buffer(in, &ptr, &cLen, strip, qual, loss); err != 0 {
return nil, handleSaveBufferError(ptr)
p := C.struct_SaveParams{
inputImage: in,
outputFormat: C.TIFF,
stripMetadata: C.int(boolToInt(params.StripMetadata)),
quality: C.int(params.Quality),
tiffCompression: C.VipsForeignTiffCompression(params.Compression),
}

return toBuff(ptr, cLen), nil
return vipsSaveToBuffer(p)
}

func vipsSaveHEIFToBuffer(in *C.VipsImage, quality int, lossless bool) ([]byte, error) {
func vipsSaveHEIFToBuffer(in *C.VipsImage, params HeifExportParams) ([]byte, error) {
incOpCounter("save_heif_buffer")
var ptr unsafe.Pointer
cLen := C.size_t(0)

qual := C.int(quality)
loss := C.int(boolToInt(lossless))

if err := C.save_heif_buffer(in, &ptr, &cLen, qual, loss); err != 0 {
return nil, handleSaveBufferError(ptr)
p := C.struct_SaveParams{
inputImage: in,
outputFormat: C.HEIF,
quality: C.int(params.Quality),
heifLossless: C.int(boolToInt(params.Lossless)),
}

return toBuff(ptr, cLen), nil
return vipsSaveToBuffer(p)
}

func vipsSaveJPEGToBuffer(in *C.VipsImage, quality int, stripMetadata, interlaced bool) ([]byte, error) {
func vipsSaveJPEGToBuffer(in *C.VipsImage, params JpegExportParams) ([]byte, error) {
incOpCounter("save_jpeg_buffer")
var ptr unsafe.Pointer
cLen := C.size_t(0)

strip := C.int(boolToInt(stripMetadata))
qual := C.int(quality)
inter := C.int(boolToInt(interlaced))

if err := C.save_jpeg_buffer(in, &ptr, &cLen, strip, qual, inter); err != 0 {
return nil, handleSaveBufferError(ptr)
p := C.struct_SaveParams{
inputImage: in,
outputFormat: C.JPEG,
stripMetadata: C.int(boolToInt(params.StripMetadata)),
quality: C.int(params.Quality),
interlace: C.int(boolToInt(params.Interlace)),
}

return toBuff(ptr, cLen), nil
return vipsSaveToBuffer(p)
}

func toBuff(ptr unsafe.Pointer, cLen C.size_t) []byte {
buf := C.GoBytes(ptr, C.int(cLen))
gFreePointer(ptr)
func vipsSaveToBuffer(params C.struct_SaveParams) ([]byte, error) {
if err := C.save_to_buffer(&params); err != 0 {
return nil, handleSaveBufferError(params.outputBuffer)
}

buf := C.GoBytes(params.outputBuffer, C.int(params.outputLen))
defer gFreePointer(params.outputBuffer)

return buf
return buf, nil
}
Loading

0 comments on commit b2f6b21

Please sign in to comment.