Skip to content

Commit abf7e45

Browse files
authored
Add the ability to write Devices in Rust (#105)
* Start implementing custom devices * Add clip functions to custom device * Add more functions to custom device * Rename to NativeDevice * Add render_flags and layer * Replace `mem::forget` with `ManuallyDrop` * Add structure and metatext * Start implementing tests * Add safety comment and error handling * Simplify function requirements * Remove unused import
1 parent 0911227 commit abf7e45

File tree

6 files changed

+1321
-30
lines changed

6 files changed

+1321
-30
lines changed

mupdf-sys/build.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -118,8 +118,8 @@ fn build_libmupdf() {
118118
];
119119

120120
// this may be unused if none of the features below are enabled
121-
#[allow(unused_variables)]
122-
let add_lib = |cflags_name: &'static str, pkgcfg_name: &'static str| {
121+
#[allow(unused_variables, unused_mut)]
122+
let mut add_lib = |cflags_name: &'static str, pkgcfg_name: &'static str| {
123123
make_flags.push(format!(
124124
"SYS_{cflags_name}_CFLAGS={}",
125125
pkg_config::probe_library(pkgcfg_name)

mupdf-sys/src/lib.rs

+46
Original file line numberDiff line numberDiff line change
@@ -4,3 +4,49 @@
44
#![allow(clippy::all)]
55

66
include!(concat!(env!("OUT_DIR"), "/bindings.rs"));
7+
8+
use core::ffi::{c_int, CStr};
9+
10+
/// This function allocates a new device and returns a pointer to it if no error occured. For the
11+
/// required structure of `T` check the example below. If an error occurs the pointer `errptr` points
12+
/// to will be set to to a pointer pointing to the error and null returned from this function.
13+
///
14+
/// # Safety
15+
///
16+
/// The caller must ensure `ctx` and `errptr` to be a valid pointers.
17+
///
18+
/// It must also ensure `T` to be a type that starts with `fz_device`. Memory will be allocated for
19+
/// a new instance of `T`, but only the `fz_device` portion will be initialized. The rest is
20+
/// currently being zero-initialized, but this might change in the future.
21+
///
22+
/// # Example
23+
///
24+
/// This is how a compliant `T` might look like. The `repr(C)` is necessary as `repr(Rust)` does
25+
/// not guarantee stable field orderings.
26+
///
27+
/// ```rust
28+
/// use mupdf_sys::fz_device;
29+
///
30+
/// #[repr(C)]
31+
/// struct MyDevice {
32+
/// base: fz_device,
33+
/// foo: u32,
34+
/// }
35+
/// ```
36+
pub unsafe fn mupdf_new_derived_device<T>(
37+
ctx: *mut fz_context,
38+
label: &'static CStr,
39+
errptr: *mut *mut mupdf_error_t,
40+
) -> *mut T {
41+
let SIZE: c_int = const {
42+
if (c_int::MAX as usize) < size_of::<T>() {
43+
panic!("device too big")
44+
} else {
45+
size_of::<T>() as c_int
46+
}
47+
};
48+
49+
let device = mupdf_new_device_of_size(ctx, SIZE, errptr);
50+
let label = Memento_label(device.cast(), label.as_ptr());
51+
label.cast()
52+
}

mupdf-sys/wrapper.c

+64-2
Original file line numberDiff line numberDiff line change
@@ -2640,6 +2640,20 @@ fz_device *mupdf_new_draw_device(fz_context *ctx, fz_pixmap *pixmap, fz_irect cl
26402640
return device;
26412641
}
26422642

2643+
fz_device *mupdf_new_device_of_size(fz_context *ctx, int size, mupdf_error_t **errptr)
2644+
{
2645+
fz_device *device = NULL;
2646+
fz_try(ctx)
2647+
{
2648+
device = fz_new_device_of_size(ctx, size);
2649+
}
2650+
fz_catch(ctx)
2651+
{
2652+
mupdf_save_error(ctx, errptr);
2653+
}
2654+
return device;
2655+
}
2656+
26432657
fz_device *mupdf_new_display_list_device(fz_context *ctx, fz_display_list *list, mupdf_error_t **errptr)
26442658
{
26452659
fz_device *device = NULL;
@@ -2863,6 +2877,54 @@ void mupdf_end_layer(fz_context *ctx, fz_device *device, mupdf_error_t **errptr)
28632877
}
28642878
}
28652879

2880+
void mupdf_begin_structure(fz_context *ctx, fz_device *device, fz_structure standard, const char *raw, int idx, mupdf_error_t **errptr)
2881+
{
2882+
fz_try(ctx)
2883+
{
2884+
fz_begin_structure(ctx, device, standard, raw, idx);
2885+
}
2886+
fz_catch(ctx)
2887+
{
2888+
mupdf_save_error(ctx, errptr);
2889+
}
2890+
}
2891+
2892+
void mupdf_end_structure(fz_context *ctx, fz_device *device, mupdf_error_t **errptr)
2893+
{
2894+
fz_try(ctx)
2895+
{
2896+
fz_end_structure(ctx, device);
2897+
}
2898+
fz_catch(ctx)
2899+
{
2900+
mupdf_save_error(ctx, errptr);
2901+
}
2902+
}
2903+
2904+
void mupdf_begin_metatext(fz_context *ctx, fz_device *device, fz_metatext meta, const char *text, mupdf_error_t **errptr)
2905+
{
2906+
fz_try(ctx)
2907+
{
2908+
fz_begin_metatext(ctx, device, meta, text);
2909+
}
2910+
fz_catch(ctx)
2911+
{
2912+
mupdf_save_error(ctx, errptr);
2913+
}
2914+
}
2915+
2916+
void mupdf_end_metatext(fz_context *ctx, fz_device *device, mupdf_error_t **errptr)
2917+
{
2918+
fz_try(ctx)
2919+
{
2920+
fz_end_metatext(ctx, device);
2921+
}
2922+
fz_catch(ctx)
2923+
{
2924+
mupdf_save_error(ctx, errptr);
2925+
}
2926+
}
2927+
28662928
void mupdf_begin_mask(fz_context *ctx, fz_device *device, fz_rect area, bool luminosity, fz_colorspace *cs, const float *color, fz_color_params cp, mupdf_error_t **errptr)
28672929
{
28682930
fz_try(ctx)
@@ -2875,11 +2937,11 @@ void mupdf_begin_mask(fz_context *ctx, fz_device *device, fz_rect area, bool lum
28752937
}
28762938
}
28772939

2878-
void mupdf_end_mask(fz_context *ctx, fz_device *device, mupdf_error_t **errptr)
2940+
void mupdf_end_mask(fz_context *ctx, fz_device *device, fz_function *fn, mupdf_error_t **errptr)
28792941
{
28802942
fz_try(ctx)
28812943
{
2882-
fz_end_mask(ctx, device);
2944+
fz_end_mask_tr(ctx, device, fn);
28832945
}
28842946
fz_catch(ctx)
28852947
{

0 commit comments

Comments
 (0)