#![no_std]
extern crate libc;
extern crate zstd_sys;
#[cfg(feature = "std")]
extern crate std;
#[cfg(feature = "std")]
use std::os::raw::{c_int, c_ulonglong, c_void};
#[cfg(not(feature = "std"))]
use libc::{c_int, c_ulonglong, c_void};
use core::marker::PhantomData;
use core::ops::Deref;
use core::ops::DerefMut;
use core::str;
pub const VERSION_MAJOR: u32 = zstd_sys::ZSTD_VERSION_MAJOR;
pub const VERSION_MINOR: u32 = zstd_sys::ZSTD_VERSION_MINOR;
pub const VERSION_RELEASE: u32 = zstd_sys::ZSTD_VERSION_RELEASE;
pub const VERSION_NUMBER: u32 = zstd_sys::ZSTD_VERSION_NUMBER;
pub const CLEVEL_DEFAULT: i32 = zstd_sys::ZSTD_CLEVEL_DEFAULT as i32;
pub const CONTENTSIZE_UNKNOWN: u64 = zstd_sys::ZSTD_CONTENTSIZE_UNKNOWN as u64;
pub const CONTENTSIZE_ERROR: u64 = zstd_sys::ZSTD_CONTENTSIZE_ERROR as u64;
#[cfg(feature = "experimental")]
pub const MAGICNUMBER: u32 = zstd_sys::ZSTD_MAGICNUMBER;
#[cfg(feature = "experimental")]
pub const MAGIC_DICTIONARY: u32 = zstd_sys::ZSTD_MAGIC_DICTIONARY;
#[cfg(feature = "experimental")]
pub const MAGIC_SKIPPABLE_START: u32 = zstd_sys::ZSTD_MAGIC_SKIPPABLE_START;
#[cfg(feature = "experimental")]
pub const BLOCKSIZELOG_MAX: u32 = zstd_sys::ZSTD_BLOCKSIZELOG_MAX;
#[cfg(feature = "experimental")]
pub const BLOCKSIZE_MAX: u32 = zstd_sys::ZSTD_BLOCKSIZE_MAX;
#[cfg(feature = "experimental")]
pub const WINDOWLOG_MAX_32: u32 = zstd_sys::ZSTD_WINDOWLOG_MAX_32;
#[cfg(feature = "experimental")]
pub const WINDOWLOG_MAX_64: u32 = zstd_sys::ZSTD_WINDOWLOG_MAX_64;
#[cfg(feature = "experimental")]
pub const WINDOWLOG_MIN: u32 = zstd_sys::ZSTD_WINDOWLOG_MIN;
#[cfg(feature = "experimental")]
pub const HASHLOG_MIN: u32 = zstd_sys::ZSTD_HASHLOG_MIN;
#[cfg(feature = "experimental")]
pub const CHAINLOG_MAX_32: u32 = zstd_sys::ZSTD_CHAINLOG_MAX_32;
#[cfg(feature = "experimental")]
pub const CHAINLOG_MAX_64: u32 = zstd_sys::ZSTD_CHAINLOG_MAX_64;
#[cfg(feature = "experimental")]
pub const CHAINLOG_MIN: u32 = zstd_sys::ZSTD_CHAINLOG_MIN;
#[cfg(feature = "experimental")]
pub const HASHLOG3_MAX: u32 = zstd_sys::ZSTD_HASHLOG3_MAX;
#[cfg(feature = "experimental")]
pub const SEARCHLOG_MIN: u32 = zstd_sys::ZSTD_SEARCHLOG_MIN;
#[cfg(feature = "experimental")]
pub const TARGETLENGTH_MAX: u32 = zstd_sys::ZSTD_TARGETLENGTH_MAX;
#[cfg(feature = "experimental")]
pub const TARGETLENGTH_MIN: u32 = zstd_sys::ZSTD_TARGETLENGTH_MIN;
#[cfg(feature = "experimental")]
pub const LDM_MINMATCH_MAX: u32 = zstd_sys::ZSTD_LDM_MINMATCH_MAX;
#[cfg(feature = "experimental")]
pub const LDM_MINMATCH_MIN: u32 = zstd_sys::ZSTD_LDM_MINMATCH_MIN;
#[cfg(feature = "experimental")]
pub const LDM_BUCKETSIZELOG_MAX: u32 = zstd_sys::ZSTD_LDM_BUCKETSIZELOG_MAX;
#[cfg(feature = "experimental")]
pub const FRAMEHEADERSIZE_PREFIX: u32 = zstd_sys::ZSTD_FRAMEHEADERSIZE_PREFIX;
#[cfg(feature = "experimental")]
pub const FRAMEHEADERSIZE_MIN: u32 = zstd_sys::ZSTD_FRAMEHEADERSIZE_MIN;
#[cfg(feature = "experimental")]
pub const FRAMEHEADERSIZE_MAX: u32 = zstd_sys::ZSTD_FRAMEHEADERSIZE_MAX;
type SafeResult = Result<usize, usize>;
fn is_error(code: usize) -> bool {
unsafe { zstd_sys::ZSTD_isError(code) != 0 }
}
fn parse_code(code: usize) -> SafeResult {
if !is_error(code) {
Ok(code)
} else {
Err(code)
}
}
fn ptr_void(src: &[u8]) -> *const c_void {
src.as_ptr() as *const c_void
}
fn ptr_mut_void(dst: &mut [u8]) -> *mut c_void {
dst.as_mut_ptr() as *mut c_void
}
pub fn version_number() -> u32 {
unsafe { zstd_sys::ZSTD_versionNumber() as u32 }
}
pub fn compress(
dst: &mut [u8],
src: &[u8],
compression_level: i32,
) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_compress(
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
compression_level,
)
};
parse_code(code)
}
pub fn decompress(dst: &mut [u8], src: &[u8]) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_decompress(
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
)
};
parse_code(code)
}
pub fn get_decompressed_size(src: &[u8]) -> u64 {
unsafe {
zstd_sys::ZSTD_getDecompressedSize(ptr_void(src), src.len()) as u64
}
}
pub fn max_clevel() -> i32 {
unsafe { zstd_sys::ZSTD_maxCLevel() as i32 }
}
pub fn compress_bound(src_size: usize) -> usize {
unsafe { zstd_sys::ZSTD_compressBound(src_size) }
}
pub struct CCtx(*mut zstd_sys::ZSTD_CCtx);
impl Default for CCtx {
fn default() -> Self {
create_cctx()
}
}
pub fn create_cctx() -> CCtx {
CCtx(unsafe { zstd_sys::ZSTD_createCCtx() })
}
impl Drop for CCtx {
fn drop(&mut self) {
unsafe {
zstd_sys::ZSTD_freeCCtx(self.0);
}
}
}
unsafe impl Send for CCtx {}
#[cfg(not(feature = "std"))]
pub fn get_error_name(code: usize) -> &'static str {
unsafe {
let name = zstd_sys::ZSTD_getErrorName(code);
let len = libc::strlen(name);
let slice = core::slice::from_raw_parts(name as *mut u8, len);
str::from_utf8(slice).expect("bad error message from zstd")
}
}
#[cfg(feature = "std")]
pub fn get_error_name(code: usize) -> &'static str {
unsafe {
let name = zstd_sys::ZSTD_getErrorName(code);
std::ffi::CStr::from_ptr(name)
.to_str()
.expect("bad error message from zstd")
}
}
pub fn compress_cctx(
ctx: &mut CCtx,
dst: &mut [u8],
src: &[u8],
compression_level: i32,
) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_compressCCtx(
ctx.0,
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
compression_level,
)
};
parse_code(code)
}
pub struct DCtx(*mut zstd_sys::ZSTD_DCtx);
impl Default for DCtx {
fn default() -> Self {
create_dctx()
}
}
pub fn create_dctx() -> DCtx {
DCtx(unsafe { zstd_sys::ZSTD_createDCtx() })
}
impl Drop for DCtx {
fn drop(&mut self) {
unsafe {
zstd_sys::ZSTD_freeDCtx(self.0);
}
}
}
unsafe impl Send for DCtx {}
pub fn decompress_dctx(
ctx: &mut DCtx,
dst: &mut [u8],
src: &[u8],
) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_decompressDCtx(
ctx.0,
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
)
};
parse_code(code)
}
pub fn compress_using_dict(
ctx: &mut CCtx,
dst: &mut [u8],
src: &[u8],
dict: &[u8],
compression_level: i32,
) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_compress_usingDict(
ctx.0,
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
ptr_void(dict),
dict.len(),
compression_level,
)
};
parse_code(code)
}
pub fn decompress_using_dict(
dctx: &mut DCtx,
dst: &mut [u8],
src: &[u8],
dict: &[u8],
) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_decompress_usingDict(
dctx.0,
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
ptr_void(dict),
dict.len(),
)
};
parse_code(code)
}
pub struct CDict<'a>(*mut zstd_sys::ZSTD_CDict, PhantomData<&'a ()>);
pub fn create_cdict(
dict_buffer: &[u8],
compression_level: i32,
) -> CDict<'static> {
CDict(
unsafe {
zstd_sys::ZSTD_createCDict(
ptr_void(dict_buffer),
dict_buffer.len(),
compression_level,
)
},
PhantomData,
)
}
impl<'a> Drop for CDict<'a> {
fn drop(&mut self) {
unsafe {
zstd_sys::ZSTD_freeCDict(self.0);
}
}
}
unsafe impl<'a> Send for CDict<'a> {}
unsafe impl<'a> Sync for CDict<'a> {}
pub fn compress_using_cdict(
cctx: &mut CCtx,
dst: &mut [u8],
src: &[u8],
cdict: &CDict,
) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_compress_usingCDict(
cctx.0,
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
cdict.0,
)
};
parse_code(code)
}
pub struct DDict<'a>(*mut zstd_sys::ZSTD_DDict, PhantomData<&'a ()>);
pub fn create_ddict(dict_buffer: &[u8]) -> DDict<'static> {
DDict(
unsafe {
zstd_sys::ZSTD_createDDict(
ptr_void(dict_buffer),
dict_buffer.len(),
)
},
PhantomData,
)
}
impl<'a> Drop for DDict<'a> {
fn drop(&mut self) {
unsafe {
zstd_sys::ZSTD_freeDDict(self.0);
}
}
}
unsafe impl<'a> Send for DDict<'a> {}
unsafe impl<'a> Sync for DDict<'a> {}
pub fn decompress_using_ddict(
dctx: &mut DCtx,
dst: &mut [u8],
src: &[u8],
ddict: &DDict,
) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_decompress_usingDDict(
dctx.0,
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
ddict.0,
)
};
parse_code(code)
}
pub type CStream = CCtx;
pub fn create_cstream() -> CStream {
CCtx(unsafe { zstd_sys::ZSTD_createCStream() })
}
pub fn init_cstream(zcs: &mut CStream, compression_level: i32) -> usize {
unsafe { zstd_sys::ZSTD_initCStream(zcs.0, compression_level) }
}
#[derive(Debug)]
pub struct InBuffer<'a> {
pub src: &'a [u8],
pub pos: usize,
}
#[derive(Debug)]
pub struct OutBuffer<'a> {
pub dst: &'a mut [u8],
pub pos: usize,
}
fn ptr_mut<B>(ptr_void: &mut B) -> *mut B {
ptr_void as *mut B
}
struct OutBufferWrapper<'a, 'b: 'a> {
buf: zstd_sys::ZSTD_outBuffer,
parent: &'a mut OutBuffer<'b>,
}
impl<'a, 'b: 'a> Deref for OutBufferWrapper<'a, 'b> {
type Target = zstd_sys::ZSTD_outBuffer;
fn deref(&self) -> &Self::Target {
&self.buf
}
}
impl<'a, 'b: 'a> DerefMut for OutBufferWrapper<'a, 'b> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.buf
}
}
impl<'a> OutBuffer<'a> {
pub fn around(dst: &'a mut [u8]) -> Self {
OutBuffer { dst, pos: 0 }
}
fn wrap<'b>(&'b mut self) -> OutBufferWrapper<'b, 'a> {
OutBufferWrapper {
buf: zstd_sys::ZSTD_outBuffer {
dst: ptr_mut_void(self.dst),
size: self.dst.len(),
pos: self.pos,
},
parent: self,
}
}
pub fn as_slice<'b>(&'b self) -> &'a [u8]
where
'b: 'a,
{
let pos = self.pos;
&self.dst[..pos]
}
}
impl<'a, 'b> Drop for OutBufferWrapper<'a, 'b> {
fn drop(&mut self) {
self.parent.pos = self.buf.pos;
}
}
struct InBufferWrapper<'a, 'b: 'a> {
buf: zstd_sys::ZSTD_inBuffer,
parent: &'a mut InBuffer<'b>,
}
impl<'a, 'b: 'a> Deref for InBufferWrapper<'a, 'b> {
type Target = zstd_sys::ZSTD_inBuffer;
fn deref(&self) -> &Self::Target {
&self.buf
}
}
impl<'a, 'b: 'a> DerefMut for InBufferWrapper<'a, 'b> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.buf
}
}
impl<'a> InBuffer<'a> {
pub fn around(src: &'a [u8]) -> Self {
InBuffer { src, pos: 0 }
}
fn wrap<'b>(&'b mut self) -> InBufferWrapper<'b, 'a> {
InBufferWrapper {
buf: zstd_sys::ZSTD_inBuffer {
src: ptr_void(self.src),
size: self.src.len(),
pos: self.pos,
},
parent: self,
}
}
}
impl<'a, 'b> Drop for InBufferWrapper<'a, 'b> {
fn drop(&mut self) {
self.parent.pos = self.buf.pos;
}
}
pub fn compress_stream(
zcs: &mut CStream,
output: &mut OutBuffer,
input: &mut InBuffer,
) -> SafeResult {
let mut output = output.wrap();
let mut input = input.wrap();
let code = unsafe {
zstd_sys::ZSTD_compressStream(
zcs.0,
ptr_mut(&mut output),
ptr_mut(&mut input),
)
};
parse_code(code)
}
pub fn flush_stream(zcs: &mut CStream, output: &mut OutBuffer) -> SafeResult {
let mut output = output.wrap();
let code =
unsafe { zstd_sys::ZSTD_flushStream(zcs.0, ptr_mut(&mut output)) };
parse_code(code)
}
pub fn end_stream(zcs: &mut CStream, output: &mut OutBuffer) -> SafeResult {
let mut output = output.wrap();
let code =
unsafe { zstd_sys::ZSTD_endStream(zcs.0, ptr_mut(&mut output)) };
parse_code(code)
}
pub fn cstream_in_size() -> usize {
unsafe { zstd_sys::ZSTD_CStreamInSize() }
}
pub fn cstream_out_size() -> usize {
unsafe { zstd_sys::ZSTD_CStreamOutSize() }
}
pub type DStream = DCtx;
pub fn create_dstream() -> DStream {
DCtx(unsafe { zstd_sys::ZSTD_createDStream() })
}
pub fn init_dstream(zds: &mut DStream) -> usize {
unsafe { zstd_sys::ZSTD_initDStream(zds.0) }
}
pub fn decompress_stream(
zds: &mut DStream,
output: &mut OutBuffer,
input: &mut InBuffer,
) -> SafeResult {
let mut output = output.wrap();
let mut input = input.wrap();
let code = unsafe {
zstd_sys::ZSTD_decompressStream(
zds.0,
ptr_mut(&mut output),
ptr_mut(&mut input),
)
};
parse_code(code)
}
pub fn dstream_in_size() -> usize {
unsafe { zstd_sys::ZSTD_DStreamInSize() }
}
pub fn dstream_out_size() -> usize {
unsafe { zstd_sys::ZSTD_DStreamOutSize() }
}
#[cfg(feature = "experimental")]
pub fn find_frame_compressed_size(src: &[u8]) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_findFrameCompressedSize(ptr_void(src), src.len())
};
parse_code(code)
}
pub fn get_frame_content_size(src: &[u8]) -> u64 {
unsafe { zstd_sys::ZSTD_getFrameContentSize(ptr_void(src), src.len()) }
}
#[cfg(feature = "experimental")]
pub fn find_decompressed_size(src: &[u8]) -> u64 {
unsafe { zstd_sys::ZSTD_findDecompressedSize(ptr_void(src), src.len()) }
}
#[cfg(feature = "experimental")]
pub fn sizeof_cctx(cctx: &CCtx) -> usize {
unsafe { zstd_sys::ZSTD_sizeof_CCtx(cctx.0) }
}
#[cfg(feature = "experimental")]
pub fn sizeof_dctx(dctx: &DCtx) -> usize {
unsafe { zstd_sys::ZSTD_sizeof_DCtx(dctx.0) }
}
#[cfg(feature = "experimental")]
pub fn sizeof_cstream(zcs: &CStream) -> usize {
unsafe { zstd_sys::ZSTD_sizeof_CStream(zcs.0) }
}
#[cfg(feature = "experimental")]
pub fn sizeof_dstream(zds: &DStream) -> usize {
unsafe { zstd_sys::ZSTD_sizeof_DStream(zds.0) }
}
#[cfg(feature = "experimental")]
pub fn sizeof_cdict(cdict: &CDict) -> usize {
unsafe { zstd_sys::ZSTD_sizeof_CDict(cdict.0) }
}
#[cfg(feature = "experimental")]
pub fn sizeof_ddict(ddict: &DDict) -> usize {
unsafe { zstd_sys::ZSTD_sizeof_DDict(ddict.0) }
}
#[cfg(feature = "experimental")]
pub fn create_cdict_by_reference(
dict_buffer: &[u8],
compression_level: i32,
) -> CDict {
CDict(
unsafe {
zstd_sys::ZSTD_createCDict_byReference(
ptr_void(dict_buffer),
dict_buffer.len(),
compression_level,
)
},
PhantomData,
)
}
#[cfg(feature = "experimental")]
pub fn is_frame(buffer: &[u8]) -> u32 {
unsafe { zstd_sys::ZSTD_isFrame(ptr_void(buffer), buffer.len()) as u32 }
}
#[cfg(feature = "experimental")]
pub fn create_ddict_by_reference(dict_buffer: &[u8]) -> DDict {
DDict(
unsafe {
zstd_sys::ZSTD_createDDict_byReference(
ptr_void(dict_buffer),
dict_buffer.len(),
)
},
PhantomData,
)
}
#[cfg(feature = "experimental")]
pub fn get_dict_id_from_dict(dict: &[u8]) -> u32 {
unsafe {
zstd_sys::ZSTD_getDictID_fromDict(ptr_void(dict), dict.len()) as u32
}
}
#[cfg(feature = "experimental")]
pub fn get_dict_id_from_ddict(ddict: &DDict) -> u32 {
unsafe { zstd_sys::ZSTD_getDictID_fromDDict(ddict.0) as u32 }
}
#[cfg(feature = "experimental")]
pub fn get_dict_id_from_frame(src: &[u8]) -> u32 {
unsafe {
zstd_sys::ZSTD_getDictID_fromFrame(ptr_void(src), src.len()) as u32
}
}
#[cfg(feature = "experimental")]
pub fn init_cstream_src_size(
zcs: &mut CStream,
compression_level: i32,
pledged_src_size: u64,
) -> usize {
unsafe {
zstd_sys::ZSTD_initCStream_srcSize(
zcs.0,
compression_level as c_int,
pledged_src_size as c_ulonglong,
)
}
}
#[cfg(feature = "experimental")]
pub fn init_cstream_using_dict(
zcs: &mut CStream,
dict: &[u8],
compression_level: i32,
) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_initCStream_usingDict(
zcs.0,
ptr_void(dict),
dict.len(),
compression_level,
)
};
parse_code(code)
}
#[cfg(feature = "experimental")]
pub fn init_cstream_using_cdict(
zcs: &mut CStream,
cdict: &CDict,
) -> SafeResult {
let code =
unsafe { zstd_sys::ZSTD_initCStream_usingCDict(zcs.0, cdict.0) };
parse_code(code)
}
#[cfg(feature = "experimental")]
pub fn reset_cstream(zcs: &mut CStream, pledged_src_size: u64) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_resetCStream(zcs.0, pledged_src_size as c_ulonglong)
};
parse_code(code)
}
#[cfg(feature = "experimental")]
pub fn init_dstream_using_dict(zds: &mut DStream, dict: &[u8]) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_initDStream_usingDict(zds.0, ptr_void(dict), dict.len())
};
parse_code(code)
}
#[cfg(feature = "experimental")]
pub fn init_dstream_using_ddict(
zds: &mut DStream,
ddict: &DDict,
) -> SafeResult {
let code =
unsafe { zstd_sys::ZSTD_initDStream_usingDDict(zds.0, ddict.0) };
parse_code(code)
}
#[cfg(feature = "experimental")]
pub fn reset_dstream(zds: &mut DStream) -> SafeResult {
let code = unsafe { zstd_sys::ZSTD_resetDStream(zds.0) };
parse_code(code)
}
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
pub enum FrameFormat {
One,
Magicless,
}
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
pub enum CParameter {
Format(FrameFormat),
CompressionLevel(i32),
WindowLog(u32),
HashLog(u32),
ChainLog(u32),
SearchLog(u32),
MinMatch(u32),
TargetLength(u32),
ContentSizeFlag(bool),
ChecksumFlag(bool),
DictIdFlag(bool),
#[cfg(feature = "zstdmt")]
ThreadCount(u32),
#[cfg(feature = "zstdmt")]
JobSize(u32),
#[cfg(feature = "zstdmt")]
OverlapSizeLog(u32),
}
#[cfg(feature = "experimental")]
pub fn cctx_set_parameter(cctx: &mut CCtx, param: CParameter) -> SafeResult {
use zstd_sys::ZSTD_cParameter;
use zstd_sys::ZSTD_cParameter::ZSTD_c_experimentalParam2 as ZSTD_c_format;
use zstd_sys::ZSTD_format_e;
use CParameter::*;
let (param, value) = match param {
Format(FrameFormat::One) => {
(ZSTD_c_format, ZSTD_format_e::ZSTD_f_zstd1 as c_int)
}
Format(FrameFormat::Magicless) => (
ZSTD_c_format,
ZSTD_format_e::ZSTD_f_zstd1_magicless as c_int,
),
CompressionLevel(level) => {
(ZSTD_cParameter::ZSTD_c_compressionLevel, level)
}
WindowLog(value) => {
(ZSTD_cParameter::ZSTD_c_windowLog, value as c_int)
}
HashLog(value) => (ZSTD_cParameter::ZSTD_c_hashLog, value as c_int),
ChainLog(value) => (ZSTD_cParameter::ZSTD_c_chainLog, value as c_int),
SearchLog(value) => {
(ZSTD_cParameter::ZSTD_c_searchLog, value as c_int)
}
MinMatch(value) => (ZSTD_cParameter::ZSTD_c_minMatch, value as c_int),
TargetLength(value) => {
(ZSTD_cParameter::ZSTD_c_targetLength, value as c_int)
}
ContentSizeFlag(flag) => (
ZSTD_cParameter::ZSTD_c_contentSizeFlag,
if flag { 1 } else { 0 },
),
ChecksumFlag(flag) => (
ZSTD_cParameter::ZSTD_c_checksumFlag,
if flag { 1 } else { 0 },
),
DictIdFlag(flag) => {
(ZSTD_cParameter::ZSTD_c_dictIDFlag, if flag { 1 } else { 0 })
}
#[cfg(feature = "zstdmt")]
ThreadCount(value) => {
(ZSTD_cParameter::ZSTD_c_nbWorkers, value as c_int)
}
#[cfg(feature = "zstdmt")]
JobSize(value) => (ZSTD_cParameter::ZSTD_c_jobSize, value as c_int),
#[cfg(feature = "zstdmt")]
OverlapSizeLog(value) => {
(ZSTD_cParameter::ZSTD_c_overlapLog, value as c_int)
}
};
let code =
unsafe { zstd_sys::ZSTD_CCtx_setParameter(cctx.0, param, value) };
parse_code(code)
}
pub fn train_from_buffer(
dict_buffer: &mut [u8],
samples_buffer: &[u8],
samples_sizes: &[usize],
) -> SafeResult {
assert_eq!(samples_buffer.len(), samples_sizes.iter().sum());
let code = unsafe {
zstd_sys::ZDICT_trainFromBuffer(
ptr_mut_void(dict_buffer),
dict_buffer.len(),
ptr_void(samples_buffer),
samples_sizes.as_ptr(),
samples_sizes.len() as u32,
)
};
parse_code(code)
}
#[cfg(feature = "experimental")]
pub fn get_block_size(cctx: &mut CCtx) -> usize {
unsafe { zstd_sys::ZSTD_getBlockSize(cctx.0) }
}
#[cfg(feature = "experimental")]
pub fn compress_block(
cctx: &mut CCtx,
dst: &mut [u8],
src: &[u8],
) -> SafeResult {
let code = unsafe {
zstd_sys::ZSTD_compressBlock(
cctx.0,
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
)
};
parse_code(code)
}
#[cfg(feature = "experimental")]
pub fn decompress_block(dctx: &mut DCtx, dst: &mut [u8], src: &[u8]) -> usize {
unsafe {
zstd_sys::ZSTD_decompressBlock(
dctx.0,
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
)
}
}
#[cfg(feature = "experimental")]
pub fn insert_block(dctx: &mut DCtx, block: &[u8]) -> usize {
unsafe { zstd_sys::ZSTD_insertBlock(dctx.0, ptr_void(block), block.len()) }
}
#[cfg(feature = "zstdmt")]
pub mod mt {
use super::c_ulonglong;
use super::parse_code;
use super::zstd_sys;
use super::SafeResult;
use super::{ptr_mut, ptr_mut_void, ptr_void, InBuffer, OutBuffer};
pub struct CCtx(*mut zstd_sys::ZSTDMT_CCtx);
pub fn create_cctx(n_threads: u32) -> CCtx {
CCtx(unsafe { zstd_sys::ZSTDMT_createCCtx(n_threads) })
}
impl Drop for CCtx {
fn drop(&mut self) {
unsafe {
zstd_sys::ZSTDMT_freeCCtx(self.0);
}
}
}
pub fn sizeof_cctx(cctx: &CCtx) -> usize {
unsafe { zstd_sys::ZSTDMT_sizeof_CCtx(cctx.0) }
}
pub fn compress_cctx(
mtctx: &mut CCtx,
dst: &mut [u8],
src: &[u8],
compression_level: i32,
) -> usize {
unsafe {
zstd_sys::ZSTDMT_compressCCtx(
mtctx.0,
ptr_mut_void(dst),
dst.len(),
ptr_void(src),
src.len(),
compression_level,
)
}
}
pub fn init_cstream(mtctx: &mut CCtx, compression_level: i32) -> usize {
unsafe { zstd_sys::ZSTDMT_initCStream(mtctx.0, compression_level) }
}
pub fn reset_cstream(mtctx: &mut CCtx, pledged_src_size: u64) -> usize {
unsafe {
zstd_sys::ZSTDMT_resetCStream(
mtctx.0,
pledged_src_size as c_ulonglong,
)
}
}
pub fn compress_stream(
mtctx: &mut CCtx,
output: &mut OutBuffer,
input: &mut InBuffer,
) -> SafeResult {
let mut output = output.wrap();
let mut input = input.wrap();
let code = unsafe {
zstd_sys::ZSTDMT_compressStream(
mtctx.0,
ptr_mut(&mut output),
ptr_mut(&mut input),
)
};
parse_code(code)
}
pub fn flush_stream(
mtctx: &mut CCtx,
output: &mut OutBuffer,
) -> SafeResult {
let mut output = output.wrap();
let code = unsafe {
zstd_sys::ZSTDMT_flushStream(mtctx.0, ptr_mut(&mut output))
};
parse_code(code)
}
pub fn end_stream(mtctx: &mut CCtx, output: &mut OutBuffer) -> SafeResult {
let mut output = output.wrap();
let code = unsafe {
zstd_sys::ZSTDMT_endStream(mtctx.0, ptr_mut(&mut output))
};
parse_code(code)
}
}