// Copyright 2020 Tyler Neely // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. use std::ffi::CString; use std::slice; use libc::{c_char, c_uchar, c_void, size_t}; use crate::{ffi, ffi_util::CStrLike}; /// A `SliceTransform` is a generic pluggable way of transforming one string /// to another. Its primary use-case is in configuring rocksdb /// to store prefix blooms by setting prefix_extractor in /// ColumnFamilyOptions. pub struct SliceTransform { pub inner: *mut ffi::rocksdb_slicetransform_t, } // NB we intentionally don't implement a Drop that passes // through to rocksdb_slicetransform_destroy because // this is currently only used (to my knowledge) // by people passing it as a prefix extractor when // opening a DB. impl SliceTransform { pub fn create( name: impl CStrLike, transform_fn: TransformFn, in_domain_fn: Option, ) -> SliceTransform { let cb = Box::into_raw(Box::new(TransformCallback { name: name.into_c_string().unwrap(), transform_fn, in_domain_fn, })); let st = unsafe { ffi::rocksdb_slicetransform_create( cb as *mut c_void, Some(slice_transform_destructor_callback), Some(transform_callback), Some(in_domain_callback), // this None points to the deprecated InRange callback None, Some(slice_transform_name_callback), ) }; SliceTransform { inner: st } } pub fn create_fixed_prefix(len: size_t) -> SliceTransform { SliceTransform { inner: unsafe { ffi::rocksdb_slicetransform_create_fixed_prefix(len) }, } } pub fn create_noop() -> SliceTransform { SliceTransform { inner: unsafe { ffi::rocksdb_slicetransform_create_noop() }, } } } pub type TransformFn<'a> = fn(&'a [u8]) -> &'a [u8]; pub type InDomainFn = fn(&[u8]) -> bool; pub struct TransformCallback<'a> { pub name: CString, pub transform_fn: TransformFn<'a>, pub in_domain_fn: Option, } pub unsafe extern "C" fn slice_transform_destructor_callback(raw_cb: *mut c_void) { drop(Box::from_raw(raw_cb as *mut TransformCallback)); } pub unsafe extern "C" fn slice_transform_name_callback(raw_cb: *mut c_void) -> *const c_char { let cb = &mut *(raw_cb as *mut TransformCallback); cb.name.as_ptr() } pub unsafe extern "C" fn transform_callback( raw_cb: *mut c_void, raw_key: *const c_char, key_len: size_t, dst_length: *mut size_t, ) -> *mut c_char { let cb = &mut *(raw_cb as *mut TransformCallback); let key = slice::from_raw_parts(raw_key as *const u8, key_len); let prefix = (cb.transform_fn)(key); *dst_length = prefix.len() as size_t; prefix.as_ptr() as *mut c_char } pub unsafe extern "C" fn in_domain_callback( raw_cb: *mut c_void, raw_key: *const c_char, key_len: size_t, ) -> c_uchar { let cb = &mut *(raw_cb as *mut TransformCallback); let key = slice::from_raw_parts(raw_key as *const u8, key_len); c_uchar::from(cb.in_domain_fn.map_or(true, |in_domain| in_domain(key))) }