blob: d4db364d20dc9a8fc7792804a134feb151f04c97 [file] [log] [blame] [edit]
// File: basisu_wasm_api.cpp - Simplified compression API for WASM WASI modules and Python native support.
// Also useable by plain C callers.
#include "basisu_comp.h"
#include "basisu_wasm_api.h"
using namespace basisu;
static inline uint64_t wasm_offset(void* p)
{
return (uint64_t)(uintptr_t)p;
}
static inline uint8_t* wasm_ptr(uint64_t offset)
{
return (uint8_t*)(uintptr_t)offset;
}
BU_WASM_EXPORT("bu_get_version")
uint32_t bu_get_version()
{
printf("Hello from basisu_wasm_api.cpp version %u\n", BASISU_LIB_VERSION);
return BASISU_LIB_VERSION;
}
BU_WASM_EXPORT("bu_enable_debug_printf")
void bu_enable_debug_printf(uint32_t flag)
{
enable_debug_printf(flag != 0);
}
BU_WASM_EXPORT("bu_init")
void bu_init()
{
basisu_encoder_init(false, false);
}
// Memory alloc/free — stubs
BU_WASM_EXPORT("bu_alloc")
uint64_t bu_alloc(uint64_t size)
{
void* p = malloc((size_t)size);
return wasm_offset(p);
}
BU_WASM_EXPORT("bu_free")
void bu_free(uint64_t ofs)
{
free(wasm_ptr(ofs));
}
const uint32_t COMP_PARAMS_MAGIC = 0x43504D50; // "CPMP"
struct comp_params
{
uint32_t m_magic = COMP_PARAMS_MAGIC;
comp_params()
{
clear();
}
void clear()
{
assert(m_magic == COMP_PARAMS_MAGIC);
m_comp_data.clear();
m_images.clear();
m_imagesf.clear();
m_stats.clear();
}
uint8_vec m_comp_data;
basisu::vector<image> m_images;
basisu::vector<imagef> m_imagesf;
image_stats m_stats;
};
BU_WASM_EXPORT("bu_new_comp_params")
uint64_t bu_new_comp_params()
{
comp_params* p = new comp_params;
return wasm_offset(p);
}
BU_WASM_EXPORT("bu_delete_comp_params")
wasm_bool_t bu_delete_comp_params(uint64_t params_ofs)
{
comp_params* p = (comp_params*)wasm_ptr(params_ofs);
if (!p)
return false;
assert(p->m_magic == COMP_PARAMS_MAGIC);
if (p->m_magic != COMP_PARAMS_MAGIC)
return false;
delete p;
return true;
}
BU_WASM_EXPORT("bu_comp_params_get_comp_data_size")
uint64_t bu_comp_params_get_comp_data_size(uint64_t params_ofs)
{
comp_params* pParams = (comp_params*)wasm_ptr(params_ofs);
if (!pParams)
return 0;
assert(pParams->m_magic == COMP_PARAMS_MAGIC);
if (pParams->m_magic != COMP_PARAMS_MAGIC)
return 0;
return pParams->m_comp_data.size();
}
BU_WASM_EXPORT("bu_comp_params_get_comp_data_ofs")
uint64_t bu_comp_params_get_comp_data_ofs(uint64_t params_ofs)
{
comp_params* pParams = (comp_params*)wasm_ptr(params_ofs);
if (!pParams)
return 0;
assert(pParams->m_magic == COMP_PARAMS_MAGIC);
if (pParams->m_magic != COMP_PARAMS_MAGIC)
return 0;
return wasm_offset(pParams->m_comp_data.get_ptr());
}
BU_WASM_EXPORT("bu_comp_params_clear")
wasm_bool_t bu_comp_params_clear(uint64_t params_ofs)
{
comp_params* pParams = (comp_params*)wasm_ptr(params_ofs);
if (!pParams)
return false;
assert(pParams->m_magic == COMP_PARAMS_MAGIC);
if (pParams->m_magic != COMP_PARAMS_MAGIC)
return false;
pParams->clear();
return true;
}
// Caller wants to give us a LDR/SDR 32bpp RGBA mipmap level (4 bytes per pixel)
BU_WASM_EXPORT("bu_comp_params_set_image_rgba32")
wasm_bool_t bu_comp_params_set_image_rgba32(
uint64_t params_ofs,
uint32_t image_index,
uint64_t img_data_ofs,
uint32_t width, uint32_t height,
uint32_t pitch_in_bytes)
{
if ((!width) || (!height) || (!pitch_in_bytes))
return false;
comp_params* pParams = (comp_params*)wasm_ptr(params_ofs);
if (!pParams)
return false;
assert(pParams->m_magic == COMP_PARAMS_MAGIC);
if (pParams->m_magic != COMP_PARAMS_MAGIC)
return false;
const uint8_t* pImage = wasm_ptr(img_data_ofs);
if (!pImage)
return false;
const uint32_t bytes_per_pixel = sizeof(color_rgba);
if (pitch_in_bytes < width * bytes_per_pixel)
return false;
if (image_index >= pParams->m_images.size())
{
if (!pParams->m_images.try_resize(image_index + 1))
return false;
}
basisu::image& dst_img = pParams->m_images[image_index];
dst_img.resize(width, height);
if (pitch_in_bytes == width * bytes_per_pixel)
{
memcpy(dst_img.get_ptr(), pImage, pitch_in_bytes * height);
}
else
{
for (uint32_t y = 0; y < height; y++)
{
const uint8_t* pSrc_row = pImage + y * pitch_in_bytes;
uint8_t* pDst_row = (uint8_t *)&dst_img(0, y);
memcpy(pDst_row, pSrc_row, width * bytes_per_pixel);
} // y
}
return true;
}
// Caller wants to give us a float RGBA mipmap level (4*4=16 bytes per pixel)
BU_WASM_EXPORT("bu_comp_params_set_image_float_rgba")
wasm_bool_t bu_comp_params_set_image_float_rgba(
uint64_t params_ofs,
uint32_t image_index,
uint64_t img_data_ofs,
uint32_t width, uint32_t height,
uint32_t pitch_in_bytes)
{
if ((!width) || (!height) || (!pitch_in_bytes))
return false;
comp_params* pParams = (comp_params*)wasm_ptr(params_ofs);
if (!pParams)
return false;
assert(pParams->m_magic == COMP_PARAMS_MAGIC);
if (pParams->m_magic != COMP_PARAMS_MAGIC)
return false;
const uint8_t* pImage = wasm_ptr(img_data_ofs);
if (!pImage)
return false;
const uint32_t bytes_per_pixel = sizeof(float) * 4;
if (pitch_in_bytes < width * bytes_per_pixel)
return false;
if (image_index >= pParams->m_images.size())
{
if (!pParams->m_imagesf.try_resize(image_index + 1))
return false;
}
basisu::imagef& dst_img = pParams->m_imagesf[image_index];
dst_img.resize(width, height);
if (pitch_in_bytes == width * bytes_per_pixel)
{
memcpy((void *)dst_img.get_ptr(), (const void *)pImage, pitch_in_bytes * height);
}
else
{
for (uint32_t y = 0; y < height; y++)
{
const uint8_t* pSrc_row = pImage + y * pitch_in_bytes;
uint8_t* pDst_row = (uint8_t*)&dst_img(0, y);
memcpy(pDst_row, pSrc_row, width * bytes_per_pixel);
} // y
}
return true;
}
BU_WASM_EXPORT("bu_compress_texture")
wasm_bool_t bu_compress_texture(
uint64_t params_ofs,
uint32_t desired_basis_tex_format, // basis_tex_format
int quality_level, int effort_level,
uint64_t flags_and_quality, float low_level_uastc_rdo_or_dct_quality)
{
//enable_debug_printf((flags_and_quality & cFlagDebug) != 0);
comp_params* pParams = (comp_params*)wasm_ptr(params_ofs);
if (!pParams)
return false;
assert(pParams->m_magic == COMP_PARAMS_MAGIC);
if (pParams->m_magic != COMP_PARAMS_MAGIC)
return false;
pParams->m_comp_data.clear();
if (desired_basis_tex_format >= (uint32_t)basist::basis_tex_format::cTotalFormats)
return false;
if (!pParams->m_images.size() && !pParams->m_imagesf.size())
return false;
if (pParams->m_images.size() && pParams->m_imagesf.size())
return false;
size_t comp_size = 0;
void* pComp_data = basis_compress_internal(
(basist::basis_tex_format)desired_basis_tex_format,
pParams->m_images.size() ? &pParams->m_images : nullptr,
pParams->m_imagesf.size() ? &pParams->m_imagesf : nullptr,
(uint32_t)flags_and_quality,
low_level_uastc_rdo_or_dct_quality,
&comp_size,
&pParams->m_stats,
quality_level,
effort_level);
if (!pComp_data)
return false;
if (!pParams->m_comp_data.try_resize(comp_size))
{
basis_free_data(pComp_data);
return false;
}
memcpy(pParams->m_comp_data.get_ptr(), pComp_data, comp_size);
basis_free_data(pComp_data);
return true;
}