mirror of
https://github.com/ghostty-org/ghostty.git
synced 2025-04-20 00:18:53 +03:00
pkg/harfbuzz: some blob APIs
This commit is contained in:
130
pkg/harfbuzz/blob.zig
Normal file
130
pkg/harfbuzz/blob.zig
Normal file
@ -0,0 +1,130 @@
|
||||
const std = @import("std");
|
||||
const c = @import("c.zig");
|
||||
const Error = @import("errors.zig").Error;
|
||||
|
||||
/// Data type holding the memory modes available to client programs.
|
||||
///
|
||||
/// Regarding these various memory-modes:
|
||||
///
|
||||
/// - In no case shall the HarfBuzz client modify memory that is passed to
|
||||
/// HarfBuzz in a blob. If there is any such possibility,
|
||||
/// HB_MEMORY_MODE_DUPLICATE should be used such that HarfBuzz makes a
|
||||
/// copy immediately,
|
||||
///
|
||||
/// - Use HB_MEMORY_MODE_READONLY otherwise, unless you really really really
|
||||
/// know what you are doing,
|
||||
///
|
||||
/// - HB_MEMORY_MODE_WRITABLE is appropriate if you really made a copy of
|
||||
/// data solely for the purpose of passing to HarfBuzz and doing that
|
||||
/// just once (no reuse!),
|
||||
///
|
||||
/// - If the font is mmap()ed, it's okay to use
|
||||
/// HB_MEMORY_READONLY_MAY_MAKE_WRITABLE , however, using that mode
|
||||
/// correctly is very tricky. Use HB_MEMORY_MODE_READONLY instead.
|
||||
pub const MemoryMode = enum(u2) {
|
||||
/// HarfBuzz immediately makes a copy of the data.
|
||||
duplicate = c.HB_MEMORY_MODE_DUPLICATE,
|
||||
|
||||
/// HarfBuzz client will never modify the data, and HarfBuzz will never
|
||||
/// modify the data.
|
||||
readonly = c.HB_MEMORY_MODE_READONLY,
|
||||
|
||||
/// HarfBuzz client made a copy of the data solely for HarfBuzz, so
|
||||
/// HarfBuzz may modify the data.
|
||||
writable = c.HB_MEMORY_MODE_WRITABLE,
|
||||
|
||||
/// See above
|
||||
readonly_may_make_writable = c.HB_MEMORY_MODE_READONLY_MAY_MAKE_WRITABLE,
|
||||
};
|
||||
|
||||
/// Blobs wrap a chunk of binary data to handle lifecycle management of data
|
||||
/// while it is passed between client and HarfBuzz. Blobs are primarily
|
||||
/// used to create font faces, but also to access font face tables, as well as
|
||||
/// pass around other binary data.
|
||||
pub const Blob = struct {
|
||||
handle: *c.hb_blob_t,
|
||||
|
||||
/// Creates a new "blob" object wrapping data . The mode parameter is used
|
||||
/// to negotiate ownership and lifecycle of data .
|
||||
///
|
||||
/// Note that this function returns a freshly-allocated empty blob even
|
||||
/// if length is zero. This is in contrast to hb_blob_create(), which
|
||||
/// returns the singleton empty blob (as returned by hb_blob_get_empty())
|
||||
/// if length is zero.
|
||||
pub fn create(data: []const u8, mode: MemoryMode) Error!Blob {
|
||||
const handle = c.hb_blob_create_or_fail(
|
||||
data.ptr,
|
||||
@intCast(c_uint, data.len),
|
||||
@enumToInt(mode),
|
||||
null,
|
||||
null,
|
||||
) orelse return Error.HarfbuzzFailed;
|
||||
|
||||
return Blob{ .handle = handle };
|
||||
}
|
||||
|
||||
/// Decreases the reference count on blob , and if it reaches zero,
|
||||
/// destroys blob , freeing all memory, possibly calling the
|
||||
/// destroy-callback the blob was created for if it has not been
|
||||
/// called already.
|
||||
pub fn destroy(self: *Blob) void {
|
||||
c.hb_blob_destroy(self.handle);
|
||||
}
|
||||
|
||||
/// Attaches a user-data key/data pair to the specified blob.
|
||||
pub fn setUserData(
|
||||
self: Blob,
|
||||
comptime T: type,
|
||||
key: ?*anyopaque,
|
||||
ptr: ?*T,
|
||||
comptime destroycb: ?fn (?*T) callconv(.C) void,
|
||||
replace: bool,
|
||||
) bool {
|
||||
const Callback = struct {
|
||||
pub fn callback(data: ?*anyopaque) callconv(.C) void {
|
||||
@call(.{ .modifier = .always_inline }, destroycb, .{
|
||||
@ptrCast(?*T, @alignCast(@alignOf(T), data)),
|
||||
});
|
||||
}
|
||||
};
|
||||
|
||||
return c.hb_blob_set_user_data(
|
||||
self.handle,
|
||||
@ptrCast([*c]c.hb_user_data_key_t, key),
|
||||
ptr,
|
||||
if (destroycb != null) Callback.callback else null,
|
||||
if (replace) 1 else 0,
|
||||
) > 0;
|
||||
}
|
||||
|
||||
/// Fetches the user data associated with the specified key, attached to
|
||||
/// the specified font-functions structure.
|
||||
pub fn getUserData(
|
||||
self: Blob,
|
||||
comptime T: type,
|
||||
key: ?*anyopaque,
|
||||
) ?*T {
|
||||
const opt = c.hb_blob_get_user_data(
|
||||
self.handle,
|
||||
@ptrCast([*c]c.hb_user_data_key_t, key),
|
||||
);
|
||||
|
||||
if (opt) |ptr|
|
||||
return @ptrCast(?*T, @alignCast(@alignOf(T), ptr))
|
||||
else
|
||||
return null;
|
||||
}
|
||||
};
|
||||
|
||||
test {
|
||||
const testing = std.testing;
|
||||
|
||||
const data = "hello";
|
||||
var blob = try Blob.create(data, .readonly);
|
||||
defer blob.destroy();
|
||||
|
||||
var userdata: u8 = 127;
|
||||
var key: u8 = 0;
|
||||
try testing.expect(blob.setUserData(u8, &key, &userdata, null, false));
|
||||
try testing.expect(blob.getUserData(u8, &key).?.* == 127);
|
||||
}
|
5
pkg/harfbuzz/errors.zig
Normal file
5
pkg/harfbuzz/errors.zig
Normal file
@ -0,0 +1,5 @@
|
||||
pub const Error = error{
|
||||
/// Not very descriptive but harfbuzz doesn't actually have error
|
||||
/// codes so the best we can do!
|
||||
HarfbuzzFailed,
|
||||
};
|
@ -1,4 +1,6 @@
|
||||
pub const c = @import("c.zig");
|
||||
pub usingnamespace @import("blob.zig");
|
||||
pub usingnamespace @import("errors.zig");
|
||||
pub usingnamespace @import("version.zig");
|
||||
|
||||
test {
|
||||
|
Reference in New Issue
Block a user