1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
use std::cmp::Ordering;
use std::ffi::CString;
use std::fmt::{self, Debug, Formatter};
use std::os::raw::{c_int, c_void};
use std::slice;
use std::str::from_utf8_unchecked;
use std::sync::Arc;
use libsqlite3_sys::{sqlite3_create_collation_v2, SQLITE_OK, SQLITE_UTF8};
use crate::error::Error;
use crate::sqlite::connection::handle::ConnectionHandle;
use crate::sqlite::SqliteError;
#[derive(Clone)]
pub struct Collation {
name: Arc<str>,
collate: Arc<dyn Fn(&str, &str) -> Ordering + Send + Sync + 'static>,
call: unsafe extern "C" fn(
arg1: *mut c_void,
arg2: c_int,
arg3: *const c_void,
arg4: c_int,
arg5: *const c_void,
) -> c_int,
free: unsafe extern "C" fn(*mut c_void),
}
impl Collation {
pub fn new<N, F>(name: N, collate: F) -> Self
where
N: Into<Arc<str>>,
F: Fn(&str, &str) -> Ordering + Send + Sync + 'static,
{
unsafe extern "C" fn drop_arc_value<T>(p: *mut c_void) {
drop(Arc::from_raw(p as *mut T));
}
Collation {
name: name.into(),
collate: Arc::new(collate),
call: call_boxed_closure::<F>,
free: drop_arc_value::<F>,
}
}
pub(crate) fn create(&self, handle: &mut ConnectionHandle) -> Result<(), Error> {
let raw_f = Arc::into_raw(Arc::clone(&self.collate));
let c_name = CString::new(&*self.name)
.map_err(|_| err_protocol!("invalid collation name: {:?}", self.name))?;
let flags = SQLITE_UTF8;
let r = unsafe {
sqlite3_create_collation_v2(
handle.as_ptr(),
c_name.as_ptr(),
flags,
raw_f as *mut c_void,
Some(self.call),
Some(self.free),
)
};
if r == SQLITE_OK {
Ok(())
} else {
drop(unsafe { Arc::from_raw(raw_f) });
Err(Error::Database(Box::new(SqliteError::new(handle.as_ptr()))))
}
}
}
impl Debug for Collation {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
f.debug_struct("Collation")
.field("name", &self.name)
.finish_non_exhaustive()
}
}
pub(crate) fn create_collation<F>(
handle: &mut ConnectionHandle,
name: &str,
compare: F,
) -> Result<(), Error>
where
F: Fn(&str, &str) -> Ordering + Send + Sync + 'static,
{
unsafe extern "C" fn free_boxed_value<T>(p: *mut c_void) {
drop(Box::from_raw(p as *mut T));
}
let boxed_f: *mut F = Box::into_raw(Box::new(compare));
let c_name =
CString::new(name).map_err(|_| err_protocol!("invalid collation name: {}", name))?;
let flags = SQLITE_UTF8;
let r = unsafe {
sqlite3_create_collation_v2(
handle.as_ptr(),
c_name.as_ptr(),
flags,
boxed_f as *mut c_void,
Some(call_boxed_closure::<F>),
Some(free_boxed_value::<F>),
)
};
if r == SQLITE_OK {
Ok(())
} else {
drop(unsafe { Box::from_raw(boxed_f) });
Err(Error::Database(Box::new(SqliteError::new(handle.as_ptr()))))
}
}
unsafe extern "C" fn call_boxed_closure<C>(
data: *mut c_void,
left_len: c_int,
left_ptr: *const c_void,
right_len: c_int,
right_ptr: *const c_void,
) -> c_int
where
C: Fn(&str, &str) -> Ordering,
{
let boxed_f: *mut C = data as *mut C;
debug_assert!(!boxed_f.is_null());
let s1 = {
let c_slice = slice::from_raw_parts(left_ptr as *const u8, left_len as usize);
from_utf8_unchecked(c_slice)
};
let s2 = {
let c_slice = slice::from_raw_parts(right_ptr as *const u8, right_len as usize);
from_utf8_unchecked(c_slice)
};
let t = (*boxed_f)(s1, s2);
match t {
Ordering::Less => -1,
Ordering::Equal => 0,
Ordering::Greater => 1,
}
}