Skip to main content

yog_runtime/
lib.rs

1//! Yog runtime — the native library loaded by the Fabric host.
2//!
3//! Exposes JNI entry points (`Java_dev_yog_NativeBridge_*`) that the host calls,
4//! and a stable C ABI (`YogApi` / `YogServer`) that mods program against.
5//!
6//! Architecture:
7//!   - `YogServer`  — a `#[repr(C)]` table of standalone JNI-calling functions
8//!                    that mods call to mutate the world.
9//!   - `YogApi`     — a `#[repr(C)]` table of registration functions; mods call
10//!                    them inside `yog_mod_register` to subscribe to events.
11//!   - `RuntimeHandlers` — the runtime's internal event/handler storage.
12//!     Filled during `nativeInit` (write), read-only after. The scheduler sub-
13//!     state uses an inner `Mutex` for safe addition during event dispatch.
14
15use std::collections::HashMap;
16use std::num::NonZeroU32;
17use std::os::raw::c_void;
18use std::path::{Path, PathBuf};
19use std::sync::{Mutex, OnceLock};
20
21use glow::HasContext;
22use jni::objects::{JByteArray, JClass, JFloatArray, JObject, JString, JValue};
23use jni::sys::{jdouble, jfloat, jint, jstring};
24use jni::{JNIEnv, JavaVM};
25use libloading::{Library, Symbol};
26
27use yog_abi::{
28    ABI_VERSION, YogAdvancementEvent, YogAdvancementFn, YogApi, YogAttackEntityFn,
29    YogBlockBreakFn, YogBlockDef, YogBlockPos, YogChatFn, YogClientFn, YogCommandFn,
30    YogContainerCloseEvent, YogContainerCloseFn, YogContainerOpenEvent, YogContainerOpenFn,
31    YogCraftEvent, YogCraftFn, YogEntityDamageFn, YogEntityDeathFn, YogEntityInteractEvent,
32    YogEntityInteractFn, YogEntitySpawnFn, YogExplosionEvent, YogExplosionFn,
33    YogGfxApi, YogHudRenderFn, YogItemDef, YogItemPickupEvent, YogItemPickupFn,
34    YogKeyPressFn, YogKeyPressEvent, YogOwnedStr, YogPacketFn, YogPlaceBlockEvent,
35    YogPlaceBlockFn, YogPlayerDeathEvent, YogPlayerDeathFn, YogPlayerFn, YogPlayerMoveEvent,
36    YogPlayerMoveFn, YogPlayerRespawnEvent, YogPlayerRespawnFn, YogProjectileHitEvent,
37    YogProjectileHitFn, YogScheduledFn, YogScreenFn, YogServer, YogServerFn, YogStr, YogStartupGrantDef,
38    YogUseBlockFn, YogUseItemFn, YogVec3, YogWorldRenderFn,
39};
40use yog_registry::{BlockDef, FoodDef, ItemDef};
41
42// ── Static globals ────────────────────────────────────────────────────────────
43
44/// Cached JVM handle for any-thread callbacks.
45static JAVA_VM: OnceLock<JavaVM> = OnceLock::new();
46/// Loaded mod libraries — kept alive so the code pages stay mapped.
47static LOADED_MODS: Mutex<Vec<Library>> = Mutex::new(Vec::new());
48/// Stable server table (populated once in nativeInit, then read-only).
49static SERVER: OnceLock<YogServer> = OnceLock::new();
50/// All registered handlers + content (populated during mod loading, then read-only).
51static HANDLERS: OnceLock<RuntimeHandlers> = OnceLock::new();
52
53// ── OpenGL context (client-side, render thread only) ─────────────────────────
54
55struct GlCtx(glow::Context);
56unsafe impl Send for GlCtx {}
57unsafe impl Sync for GlCtx {}
58
59/// Initialized by `nativeGlInit` on the render thread.  `None` on dedicated server.
60static GL: OnceLock<GlCtx> = OnceLock::new();
61
62// Raw GL function pointers for GL_ARB_get_program_binary (not exposed by glow 0.13).
63// Captured during the glow loader callback in `nativeGlInit`.
64// `None` when the extension is unavailable (very old drivers).
65static GL_GET_PROGRAM_BINARY: OnceLock<Option<usize>> = OnceLock::new();
66static GL_PROGRAM_BINARY:     OnceLock<Option<usize>> = OnceLock::new();
67static GL_GET_PROGRAM_IV:     OnceLock<Option<usize>> = OnceLock::new();
68
69// ── Handler storage ───────────────────────────────────────────────────────────
70
71struct RuntimeHandlers {
72    block_break:        Vec<(*mut c_void, YogBlockBreakFn)>,
73    chat:               Vec<(*mut c_void, YogChatFn)>,
74    player_join:        Vec<(*mut c_void, YogPlayerFn)>,
75    player_leave:       Vec<(*mut c_void, YogPlayerFn)>,
76    use_item:           Vec<(*mut c_void, YogUseItemFn)>,
77    use_block:          Vec<(*mut c_void, YogUseBlockFn)>,
78    attack_entity:      Vec<(*mut c_void, YogAttackEntityFn)>,
79    entity_damage:      Vec<(*mut c_void, YogEntityDamageFn)>,
80    entity_death:       Vec<(*mut c_void, YogEntityDeathFn)>,
81    entity_spawn:       Vec<(*mut c_void, YogEntitySpawnFn)>,
82    player_place_block: Vec<(*mut c_void, YogPlaceBlockFn)>,
83    player_death:       Vec<(*mut c_void, YogPlayerDeathFn)>,
84    player_respawn:     Vec<(*mut c_void, YogPlayerRespawnFn)>,
85    advancement:        Vec<(*mut c_void, YogAdvancementFn)>,
86    entity_interact:    Vec<(*mut c_void, YogEntityInteractFn)>,
87    item_craft:         Vec<(*mut c_void, YogCraftFn)>,
88    explosion:          Vec<(*mut c_void, YogExplosionFn)>,
89    item_pickup:        Vec<(*mut c_void, YogItemPickupFn)>,
90    player_move:        Vec<(*mut c_void, YogPlayerMoveFn)>,
91    container_open:     Vec<(*mut c_void, YogContainerOpenFn)>,
92    container_close:    Vec<(*mut c_void, YogContainerCloseFn)>,
93    projectile_hit:     Vec<(*mut c_void, YogProjectileHitFn)>,
94    client_tick:        Vec<(*mut c_void, YogClientFn)>,
95    hud_render:         Vec<(*mut c_void, YogHudRenderFn)>,
96    world_render:       Vec<(*mut c_void, YogWorldRenderFn)>,
97    key_press:          Vec<(*mut c_void, YogKeyPressFn)>,
98    screen_open:        Vec<(*mut c_void, YogScreenFn)>,
99    screen_close:       Vec<(*mut c_void, YogScreenFn)>,
100    server_tick:        Vec<(*mut c_void, YogServerFn)>,
101    server_started:     Vec<(*mut c_void, YogServerFn)>,
102    server_stopping:    Vec<(*mut c_void, YogServerFn)>,
103    commands:           HashMap<String, (*mut c_void, YogCommandFn)>,
104    typed_schemas:      HashMap<String, String>,
105    recipes:            Vec<(String, String, String)>,
106    packets:            HashMap<String, (*mut c_void, YogPacketFn)>,
107    client_packets:     HashMap<String, (*mut c_void, YogPacketFn)>,
108    items:              Vec<ItemDef>,
109    blocks:             Vec<BlockDef>,
110    books:              HashMap<String, String>, // book_id → JSON
111    startup_grants:     Vec<yog_registry::StartupGrant>,
112    startup_granted:    Mutex<HashMap<String, bool>>,
113    scheduler:          Mutex<SchedulerState>,
114}
115
116// All fn ptrs are C-ABI; ud pointers are from Box::into_raw of Send+Sync closures.
117unsafe impl Send for RuntimeHandlers {}
118unsafe impl Sync for RuntimeHandlers {}
119
120impl RuntimeHandlers {
121    fn new() -> Self {
122        Self {
123            block_break: Vec::new(), chat: Vec::new(),
124            player_join: Vec::new(), player_leave: Vec::new(),
125            use_item: Vec::new(), use_block: Vec::new(),
126            attack_entity: Vec::new(), entity_damage: Vec::new(),
127            entity_death: Vec::new(), entity_spawn: Vec::new(),
128            player_place_block: Vec::new(),
129            player_death: Vec::new(), player_respawn: Vec::new(), advancement: Vec::new(),
130            entity_interact: Vec::new(), item_craft: Vec::new(), explosion: Vec::new(),
131            item_pickup: Vec::new(), player_move: Vec::new(),
132            container_open: Vec::new(), container_close: Vec::new(),
133            projectile_hit: Vec::new(),
134            client_tick: Vec::new(), hud_render: Vec::new(), world_render: Vec::new(),
135            key_press: Vec::new(),
136            screen_open: Vec::new(), screen_close: Vec::new(),
137            server_tick: Vec::new(), server_started: Vec::new(), server_stopping: Vec::new(),
138            commands: HashMap::new(), typed_schemas: HashMap::new(),
139            recipes: Vec::new(), packets: HashMap::new(),
140            client_packets: HashMap::new(), items: Vec::new(),
141            blocks: Vec::new(), books: HashMap::new(), startup_grants: Vec::new(),
142            startup_granted: Mutex::new(HashMap::new()),
143            scheduler: Mutex::new(SchedulerState::new()),
144        }
145    }
146}
147
148struct SchedulerState {
149    once_tasks:       Vec<OnceTask>,
150    repeating_tasks:  Vec<RepeatingTask>,
151}
152
153struct OnceTask      { delay_remaining: u64, ud: *mut c_void, f: YogScheduledFn }
154struct RepeatingTask { period: u64, ticks_left: u64, ud: *mut c_void, f: YogScheduledFn }
155
156unsafe impl Send for SchedulerState {}
157unsafe impl Sync for SchedulerState {}
158unsafe impl Send for OnceTask {}
159unsafe impl Send for RepeatingTask {}
160
161impl SchedulerState {
162    fn new() -> Self { Self { once_tasks: Vec::new(), repeating_tasks: Vec::new() } }
163}
164
165fn handlers() -> &'static RuntimeHandlers {
166    HANDLERS.get().expect("yog: nativeInit not called yet")
167}
168
169// ── JNI helpers ──────────────────────────────────────────────────────────────
170
171fn guard(label: &str, f: impl FnOnce()) {
172    if std::panic::catch_unwind(std::panic::AssertUnwindSafe(f)).is_err() {
173        yog_logging::error!("a mod panicked handling `{}` (ignored)", label);
174    }
175}
176
177macro_rules! jstr {
178    ($env:expr, $s:expr) => {
179        match $env.get_string(&$s) { Ok(s) => String::from(s), Err(_) => return }
180    };
181}
182
183/// Convert a `YogStr` into a Java String. Caller must ensure `s` is valid UTF-8.
184unsafe fn ys_to_java<'l>(env: &mut JNIEnv<'l>, s: YogStr)
185    -> Option<jni::objects::JString<'l>>
186{
187    env.new_string(s.as_str()).ok()
188}
189
190fn get_env() -> Option<jni::AttachGuard<'static>> {
191    JAVA_VM.get()?.attach_current_thread().ok()
192}
193
194// ── Free-str allocator used by YogOwnedStr ────────────────────────────────────
195
196unsafe extern "C" fn yog_free_str(ptr: *mut u8, len: u32) {
197    if !ptr.is_null() {
198        drop(Box::from_raw(std::slice::from_raw_parts_mut(ptr, len as usize)));
199    }
200}
201
202fn jstring_to_owned(env: &mut JNIEnv, obj: jni::objects::JObject) -> YogOwnedStr {
203    if obj.as_raw().is_null() { return YogOwnedStr::NONE; }
204    match env.get_string(&JString::from(obj)) {
205        Ok(s) => YogOwnedStr::from_string(String::from(s)),
206        Err(_) => YogOwnedStr::NONE,
207    }
208}
209
210// ── YogServer standalone functions (one per action) ───────────────────────────
211//
212// ctx is unused here — all state is in the JAVA_VM static.
213
214unsafe extern "C" fn srv_broadcast(_ctx: *mut c_void, msg: YogStr) {
215    let Some(mut env) = get_env() else { return };
216    if let Some(jmsg) = ys_to_java(&mut env, msg) {
217        let _ = env.call_static_method("dev/yog/NativeBridge", "broadcast",
218            "(Ljava/lang/String;)V", &[JValue::Object(&jmsg)]);
219    }
220}
221
222unsafe extern "C" fn srv_get_block(_ctx: *mut c_void, dim: YogStr, pos: YogBlockPos) -> YogOwnedStr {
223    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
224    let (Some(jd), ) = (ys_to_java(&mut env, dim),) else { return YogOwnedStr::NONE };
225    let ret = env.call_static_method("dev/yog/NativeBridge", "getBlock",
226        "(Ljava/lang/String;III)Ljava/lang/String;",
227        &[JValue::Object(&jd), JValue::Int(pos.x), JValue::Int(pos.y), JValue::Int(pos.z)]);
228    match ret.and_then(|v| v.l()) {
229        Ok(obj) => jstring_to_owned(&mut env, obj),
230        _ => YogOwnedStr::NONE,
231    }
232}
233
234unsafe extern "C" fn srv_set_block(_ctx: *mut c_void, dim: YogStr, pos: YogBlockPos, block: YogStr) -> bool {
235    let Some(mut env) = get_env() else { return false };
236    let (Some(jd), Some(jb)) = (ys_to_java(&mut env, dim), ys_to_java(&mut env, block)) else { return false };
237    env.call_static_method("dev/yog/NativeBridge", "setBlock",
238        "(Ljava/lang/String;IIILjava/lang/String;)Z",
239        &[JValue::Object(&jd), JValue::Int(pos.x), JValue::Int(pos.y), JValue::Int(pos.z), JValue::Object(&jb)])
240    .and_then(|v| v.z()).unwrap_or(false)
241}
242
243unsafe extern "C" fn srv_world_time(_ctx: *mut c_void, dim: YogStr, out: *mut i64) -> bool {
244    let Some(mut env) = get_env() else { return false };
245    let Some(jd) = ys_to_java(&mut env, dim) else { return false };
246    match env.call_static_method("dev/yog/NativeBridge", "worldTime",
247        "(Ljava/lang/String;)J", &[JValue::Object(&jd)]).and_then(|v| v.j()) {
248        Ok(v) if v != i64::MIN => { *out = v; true }
249        _ => false,
250    }
251}
252
253unsafe extern "C" fn srv_set_time(_ctx: *mut c_void, dim: YogStr, time: i64) -> bool {
254    let Some(mut env) = get_env() else { return false };
255    let Some(jd) = ys_to_java(&mut env, dim) else { return false };
256    env.call_static_method("dev/yog/NativeBridge", "worldSetTime",
257        "(Ljava/lang/String;J)Z", &[JValue::Object(&jd), JValue::Long(time)])
258    .and_then(|v| v.z()).unwrap_or(false)
259}
260
261unsafe extern "C" fn srv_is_raining(_ctx: *mut c_void, dim: YogStr) -> bool {
262    let Some(mut env) = get_env() else { return false };
263    let Some(jd) = ys_to_java(&mut env, dim) else { return false };
264    env.call_static_method("dev/yog/NativeBridge", "worldIsRaining",
265        "(Ljava/lang/String;)Z", &[JValue::Object(&jd)])
266    .and_then(|v| v.z()).unwrap_or(false)
267}
268
269unsafe extern "C" fn srv_set_weather(_ctx: *mut c_void, dim: YogStr, raining: bool, dur: i32) -> bool {
270    let Some(mut env) = get_env() else { return false };
271    let Some(jd) = ys_to_java(&mut env, dim) else { return false };
272    env.call_static_method("dev/yog/NativeBridge", "worldSetWeather",
273        "(Ljava/lang/String;ZI)Z",
274        &[JValue::Object(&jd), JValue::Bool(raining as u8), JValue::Int(dur)])
275    .and_then(|v| v.z()).unwrap_or(false)
276}
277
278unsafe extern "C" fn srv_give_item(_ctx: *mut c_void, player: YogStr, item: YogStr, count: u32) -> bool {
279    let Some(mut env) = get_env() else { return false };
280    let (Some(jp), Some(ji)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, item)) else { return false };
281    env.call_static_method("dev/yog/NativeBridge", "giveItem",
282        "(Ljava/lang/String;Ljava/lang/String;I)Z",
283        &[JValue::Object(&jp), JValue::Object(&ji), JValue::Int(count as i32)])
284    .and_then(|v| v.z()).unwrap_or(false)
285}
286
287unsafe extern "C" fn srv_player_teleport(_ctx: *mut c_void, player: YogStr, pos: YogVec3) -> bool {
288    let Some(mut env) = get_env() else { return false };
289    let Some(jp) = ys_to_java(&mut env, player) else { return false };
290    env.call_static_method("dev/yog/NativeBridge", "teleport",
291        "(Ljava/lang/String;DDD)Z",
292        &[JValue::Object(&jp), JValue::Double(pos.x), JValue::Double(pos.y), JValue::Double(pos.z)])
293    .and_then(|v| v.z()).unwrap_or(false)
294}
295
296unsafe extern "C" fn srv_send_to_player(_ctx: *mut c_void, player: YogStr, channel: YogStr, data: *const u8, len: u32) -> bool {
297    let Some(mut env) = get_env() else { return false };
298    let (Some(jp), Some(jc)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, channel)) else { return false };
299    let payload = std::slice::from_raw_parts(data, len as usize);
300    let Ok(jdata) = env.byte_array_from_slice(payload) else { return false };
301    env.call_static_method("dev/yog/NativeBridge", "sendToPlayer",
302        "(Ljava/lang/String;Ljava/lang/String;[B)Z",
303        &[JValue::Object(&jp), JValue::Object(&jc), JValue::Object(&jdata)])
304    .and_then(|v| v.z()).unwrap_or(false)
305}
306
307unsafe extern "C" fn srv_send_to_server(_ctx: *mut c_void, channel: YogStr, data: *const u8, len: u32) -> bool {
308    let Some(mut env) = get_env() else { return false };
309    let Some(jc) = ys_to_java(&mut env, channel) else { return false };
310    let payload = std::slice::from_raw_parts(data, len as usize);
311    let Ok(jdata) = env.byte_array_from_slice(payload) else { return false };
312    let result = env.call_static_method("dev/yog/YogClient", "sendToServer",
313        "(Ljava/lang/String;[B)Z", &[JValue::Object(&jc), JValue::Object(&jdata)]);
314    let _ = env.exception_clear();
315    result.and_then(|v| v.z()).unwrap_or(false)
316}
317
318unsafe extern "C" fn srv_kick_player(_ctx: *mut c_void, player: YogStr, reason: YogStr) -> bool {
319    let Some(mut env) = get_env() else { return false };
320    let (Some(jp), Some(jr)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, reason)) else { return false };
321    env.call_static_method("dev/yog/NativeBridge", "kickPlayer",
322        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&jp), JValue::Object(&jr)])
323    .and_then(|v| v.z()).unwrap_or(false)
324}
325
326unsafe extern "C" fn srv_set_gamemode(_ctx: *mut c_void, player: YogStr, mode: YogStr) -> bool {
327    let Some(mut env) = get_env() else { return false };
328    let (Some(jp), Some(jg)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, mode)) else { return false };
329    env.call_static_method("dev/yog/NativeBridge", "setGamemode",
330        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&jp), JValue::Object(&jg)])
331    .and_then(|v| v.z()).unwrap_or(false)
332}
333
334unsafe extern "C" fn srv_send_title(_ctx: *mut c_void, player: YogStr, title: YogStr, sub: YogStr, fi: i32, stay: i32, fo: i32) -> bool {
335    let Some(mut env) = get_env() else { return false };
336    let (Some(jp), Some(jt), Some(js)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, title), ys_to_java(&mut env, sub)) else { return false };
337    env.call_static_method("dev/yog/NativeBridge", "sendTitle",
338        "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;III)Z",
339        &[JValue::Object(&jp), JValue::Object(&jt), JValue::Object(&js), JValue::Int(fi), JValue::Int(stay), JValue::Int(fo)])
340    .and_then(|v| v.z()).unwrap_or(false)
341}
342
343unsafe extern "C" fn srv_send_actionbar(_ctx: *mut c_void, player: YogStr, msg: YogStr) -> bool {
344    let Some(mut env) = get_env() else { return false };
345    let (Some(jp), Some(jm)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, msg)) else { return false };
346    env.call_static_method("dev/yog/NativeBridge", "sendActionbar",
347        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&jp), JValue::Object(&jm)])
348    .and_then(|v| v.z()).unwrap_or(false)
349}
350
351unsafe extern "C" fn srv_play_sound(_ctx: *mut c_void, dim: YogStr, pos: YogVec3, sound: YogStr, vol: f32, pitch: f32) -> bool {
352    let Some(mut env) = get_env() else { return false };
353    let (Some(jd), Some(js)) = (ys_to_java(&mut env, dim), ys_to_java(&mut env, sound)) else { return false };
354    env.call_static_method("dev/yog/NativeBridge", "playSound",
355        "(Ljava/lang/String;DDDLjava/lang/String;FF)Z",
356        &[JValue::Object(&jd), JValue::Double(pos.x), JValue::Double(pos.y), JValue::Double(pos.z), JValue::Object(&js), JValue::Float(vol), JValue::Float(pitch)])
357    .and_then(|v| v.z()).unwrap_or(false)
358}
359
360unsafe extern "C" fn srv_play_sound_player(_ctx: *mut c_void, player: YogStr, sound: YogStr, vol: f32, pitch: f32) -> bool {
361    let Some(mut env) = get_env() else { return false };
362    let (Some(jp), Some(js)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, sound)) else { return false };
363    env.call_static_method("dev/yog/NativeBridge", "playSoundToPlayer",
364        "(Ljava/lang/String;Ljava/lang/String;FF)Z",
365        &[JValue::Object(&jp), JValue::Object(&js), JValue::Float(vol), JValue::Float(pitch)])
366    .and_then(|v| v.z()).unwrap_or(false)
367}
368
369unsafe extern "C" fn srv_entity_teleport(_ctx: *mut c_void, uuid: YogStr, pos: YogVec3) -> bool {
370    let Some(mut env) = get_env() else { return false };
371    let Some(ju) = ys_to_java(&mut env, uuid) else { return false };
372    env.call_static_method("dev/yog/NativeBridge", "entityTeleport",
373        "(Ljava/lang/String;DDD)Z",
374        &[JValue::Object(&ju), JValue::Double(pos.x), JValue::Double(pos.y), JValue::Double(pos.z)])
375    .and_then(|v| v.z()).unwrap_or(false)
376}
377
378unsafe extern "C" fn srv_entity_position(_ctx: *mut c_void, uuid: YogStr, out: *mut YogVec3) -> bool {
379    let Some(mut env) = get_env() else { return false };
380    let Some(ju) = ys_to_java(&mut env, uuid) else { return false };
381    let ret = env.call_static_method("dev/yog/NativeBridge", "entityPosition",
382        "(Ljava/lang/String;)Ljava/lang/String;", &[JValue::Object(&ju)]);
383    let obj = match ret.and_then(|v| v.l()) { Ok(o) => o, Err(_) => return false };
384    if obj.as_raw().is_null() { return false; }
385    let s: String = match env.get_string(&JString::from(obj)) { Ok(s) => String::from(s), Err(_) => return false };
386    let mut it = s.split('\t');
387    let (x, y, z) = (it.next(), it.next(), it.next());
388    if let (Some(x), Some(y), Some(z)) = (x.and_then(|v| v.parse().ok()), y.and_then(|v| v.parse().ok()), z.and_then(|v| v.parse().ok())) {
389        *out = YogVec3 { x, y, z }; true
390    } else { false }
391}
392
393unsafe extern "C" fn srv_entity_health(_ctx: *mut c_void, uuid: YogStr, out: *mut f32) -> bool {
394    let Some(mut env) = get_env() else { return false };
395    let Some(ju) = ys_to_java(&mut env, uuid) else { return false };
396    match env.call_static_method("dev/yog/NativeBridge", "entityHealth",
397        "(Ljava/lang/String;)D", &[JValue::Object(&ju)]).and_then(|v| v.d()) {
398        Ok(v) if !v.is_nan() => { *out = v as f32; true }
399        _ => false,
400    }
401}
402
403unsafe extern "C" fn srv_entity_set_health(_ctx: *mut c_void, uuid: YogStr, hp: f32) -> bool {
404    let Some(mut env) = get_env() else { return false };
405    let Some(ju) = ys_to_java(&mut env, uuid) else { return false };
406    env.call_static_method("dev/yog/NativeBridge", "entitySetHealth",
407        "(Ljava/lang/String;D)Z", &[JValue::Object(&ju), JValue::Double(hp as f64)])
408    .and_then(|v| v.z()).unwrap_or(false)
409}
410
411unsafe extern "C" fn srv_entity_kill(_ctx: *mut c_void, uuid: YogStr) -> bool {
412    let Some(mut env) = get_env() else { return false };
413    let Some(ju) = ys_to_java(&mut env, uuid) else { return false };
414    env.call_static_method("dev/yog/NativeBridge", "entityKill",
415        "(Ljava/lang/String;)Z", &[JValue::Object(&ju)])
416    .and_then(|v| v.z()).unwrap_or(false)
417}
418
419unsafe extern "C" fn srv_spawn_entity(_ctx: *mut c_void, type_id: YogStr, dim: YogStr, pos: YogVec3) -> YogOwnedStr {
420    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
421    let (Some(jt), Some(jd)) = (ys_to_java(&mut env, type_id), ys_to_java(&mut env, dim)) else { return YogOwnedStr::NONE };
422    let ret = env.call_static_method("dev/yog/NativeBridge", "spawnEntity",
423        "(Ljava/lang/String;Ljava/lang/String;DDD)Ljava/lang/String;",
424        &[JValue::Object(&jt), JValue::Object(&jd), JValue::Double(pos.x), JValue::Double(pos.y), JValue::Double(pos.z)]);
425    match ret.and_then(|v| v.l()) {
426        Ok(obj) => jstring_to_owned(&mut env, obj),
427        _ => YogOwnedStr::NONE,
428    }
429}
430
431unsafe extern "C" fn srv_entity_add_effect(_ctx: *mut c_void, uuid: YogStr, fx: YogStr, dur: i32, amp: u8, particles: bool) -> bool {
432    let Some(mut env) = get_env() else { return false };
433    let (Some(ju), Some(je)) = (ys_to_java(&mut env, uuid), ys_to_java(&mut env, fx)) else { return false };
434    env.call_static_method("dev/yog/NativeBridge", "entityAddEffect",
435        "(Ljava/lang/String;Ljava/lang/String;IIZ)Z",
436        &[JValue::Object(&ju), JValue::Object(&je), JValue::Int(dur), JValue::Int(amp as i32), JValue::Bool(particles as u8)])
437    .and_then(|v| v.z()).unwrap_or(false)
438}
439
440unsafe extern "C" fn srv_entity_remove_effect(_ctx: *mut c_void, uuid: YogStr, fx: YogStr) -> bool {
441    let Some(mut env) = get_env() else { return false };
442    let (Some(ju), Some(je)) = (ys_to_java(&mut env, uuid), ys_to_java(&mut env, fx)) else { return false };
443    env.call_static_method("dev/yog/NativeBridge", "entityRemoveEffect",
444        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&ju), JValue::Object(&je)])
445    .and_then(|v| v.z()).unwrap_or(false)
446}
447
448unsafe extern "C" fn srv_entity_clear_effects(_ctx: *mut c_void, uuid: YogStr) -> bool {
449    let Some(mut env) = get_env() else { return false };
450    let Some(ju) = ys_to_java(&mut env, uuid) else { return false };
451    env.call_static_method("dev/yog/NativeBridge", "entityClearEffects",
452        "(Ljava/lang/String;)Z", &[JValue::Object(&ju)])
453    .and_then(|v| v.z()).unwrap_or(false)
454}
455
456unsafe extern "C" fn srv_entity_velocity(_ctx: *mut c_void, uuid: YogStr, out: *mut YogVec3) -> bool {
457    let Some(mut env) = get_env() else { return false };
458    let Some(ju) = ys_to_java(&mut env, uuid) else { return false };
459    let ret = env.call_static_method("dev/yog/NativeBridge", "entityVelocity",
460        "(Ljava/lang/String;)Ljava/lang/String;", &[JValue::Object(&ju)]);
461    let obj = match ret.and_then(|v| v.l()) { Ok(o) => o, Err(_) => return false };
462    if obj.as_raw().is_null() { return false; }
463    let s: String = match env.get_string(&JString::from(obj)) { Ok(s) => String::from(s), Err(_) => return false };
464    let mut it = s.split('\t');
465    let (x, y, z) = (it.next(), it.next(), it.next());
466    if let (Some(x), Some(y), Some(z)) = (x.and_then(|v| v.parse().ok()), y.and_then(|v| v.parse().ok()), z.and_then(|v| v.parse().ok())) {
467        *out = YogVec3 { x, y, z }; true
468    } else { false }
469}
470
471unsafe extern "C" fn srv_entity_set_velocity(_ctx: *mut c_void, uuid: YogStr, vel: YogVec3) -> bool {
472    let Some(mut env) = get_env() else { return false };
473    let Some(ju) = ys_to_java(&mut env, uuid) else { return false };
474    env.call_static_method("dev/yog/NativeBridge", "entitySetVelocity",
475        "(Ljava/lang/String;DDD)Z",
476        &[JValue::Object(&ju), JValue::Double(vel.x), JValue::Double(vel.y), JValue::Double(vel.z)])
477    .and_then(|v| v.z()).unwrap_or(false)
478}
479
480unsafe extern "C" fn srv_entity_add_velocity(_ctx: *mut c_void, uuid: YogStr, vel: YogVec3) -> bool {
481    let Some(mut env) = get_env() else { return false };
482    let Some(ju) = ys_to_java(&mut env, uuid) else { return false };
483    env.call_static_method("dev/yog/NativeBridge", "entityAddVelocity",
484        "(Ljava/lang/String;DDD)Z",
485        &[JValue::Object(&ju), JValue::Double(vel.x), JValue::Double(vel.y), JValue::Double(vel.z)])
486    .and_then(|v| v.z()).unwrap_or(false)
487}
488
489unsafe extern "C" fn srv_has_item_tag(_ctx: *mut c_void, item: YogStr, tag: YogStr) -> bool {
490    let Some(mut env) = get_env() else { return false };
491    let (Some(ji), Some(jt)) = (ys_to_java(&mut env, item), ys_to_java(&mut env, tag)) else { return false };
492    env.call_static_method("dev/yog/NativeBridge", "hasItemTag",
493        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&ji), JValue::Object(&jt)])
494    .and_then(|v| v.z()).unwrap_or(false)
495}
496
497unsafe extern "C" fn srv_has_block_tag(_ctx: *mut c_void, block: YogStr, tag: YogStr) -> bool {
498    let Some(mut env) = get_env() else { return false };
499    let (Some(jb), Some(jt)) = (ys_to_java(&mut env, block), ys_to_java(&mut env, tag)) else { return false };
500    env.call_static_method("dev/yog/NativeBridge", "hasBlockTag",
501        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&jb), JValue::Object(&jt)])
502    .and_then(|v| v.z()).unwrap_or(false)
503}
504
505unsafe extern "C" fn srv_drop_loot(_ctx: *mut c_void, table: YogStr, dim: YogStr, pos: YogVec3) -> bool {
506    let Some(mut env) = get_env() else { return false };
507    let (Some(jt), Some(jd)) = (ys_to_java(&mut env, table), ys_to_java(&mut env, dim)) else { return false };
508    env.call_static_method("dev/yog/NativeBridge", "dropLoot",
509        "(Ljava/lang/String;Ljava/lang/String;DDD)Z",
510        &[JValue::Object(&jt), JValue::Object(&jd), JValue::Double(pos.x), JValue::Double(pos.y), JValue::Double(pos.z)])
511    .and_then(|v| v.z()).unwrap_or(false)
512}
513
514unsafe extern "C" fn srv_scoreboard_get(_ctx: *mut c_void, obj: YogStr, player: YogStr, out: *mut i32) -> bool {
515    let Some(mut env) = get_env() else { return false };
516    let (Some(jo), Some(jp)) = (ys_to_java(&mut env, obj), ys_to_java(&mut env, player)) else { return false };
517    match env.call_static_method("dev/yog/NativeBridge", "scoreboardGet",
518        "(Ljava/lang/String;Ljava/lang/String;)I", &[JValue::Object(&jo), JValue::Object(&jp)]).and_then(|v| v.i()) {
519        Ok(v) if v != i32::MIN => { *out = v; true }
520        _ => false,
521    }
522}
523
524unsafe extern "C" fn srv_scoreboard_set(_ctx: *mut c_void, obj: YogStr, player: YogStr, score: i32) -> bool {
525    let Some(mut env) = get_env() else { return false };
526    let (Some(jo), Some(jp)) = (ys_to_java(&mut env, obj), ys_to_java(&mut env, player)) else { return false };
527    env.call_static_method("dev/yog/NativeBridge", "scoreboardSet",
528        "(Ljava/lang/String;Ljava/lang/String;I)Z",
529        &[JValue::Object(&jo), JValue::Object(&jp), JValue::Int(score)])
530    .and_then(|v| v.z()).unwrap_or(false)
531}
532
533unsafe extern "C" fn srv_scoreboard_add(_ctx: *mut c_void, obj: YogStr, player: YogStr, delta: i32, out: *mut i32) -> bool {
534    let Some(mut env) = get_env() else { return false };
535    let (Some(jo), Some(jp)) = (ys_to_java(&mut env, obj), ys_to_java(&mut env, player)) else { return false };
536    match env.call_static_method("dev/yog/NativeBridge", "scoreboardAdd",
537        "(Ljava/lang/String;Ljava/lang/String;I)I",
538        &[JValue::Object(&jo), JValue::Object(&jp), JValue::Int(delta)]).and_then(|v| v.i()) {
539        Ok(v) if v != i32::MIN => { *out = v; true }
540        _ => false,
541    }
542}
543
544unsafe extern "C" fn srv_bossbar_create(_ctx: *mut c_void, id: YogStr, title: YogStr, color: YogStr, style: YogStr) -> bool {
545    let Some(mut env) = get_env() else { return false };
546    let (Some(ji), Some(jt), Some(jc), Some(js)) = (ys_to_java(&mut env, id), ys_to_java(&mut env, title), ys_to_java(&mut env, color), ys_to_java(&mut env, style)) else { return false };
547    env.call_static_method("dev/yog/NativeBridge", "bossbarCreate",
548        "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;)Z",
549        &[JValue::Object(&ji), JValue::Object(&jt), JValue::Object(&jc), JValue::Object(&js)])
550    .and_then(|v| v.z()).unwrap_or(false)
551}
552
553unsafe extern "C" fn srv_bossbar_remove(_ctx: *mut c_void, id: YogStr) -> bool {
554    let Some(mut env) = get_env() else { return false };
555    let Some(ji) = ys_to_java(&mut env, id) else { return false };
556    env.call_static_method("dev/yog/NativeBridge", "bossbarRemove",
557        "(Ljava/lang/String;)Z", &[JValue::Object(&ji)])
558    .and_then(|v| v.z()).unwrap_or(false)
559}
560
561unsafe extern "C" fn srv_bossbar_set_title(_ctx: *mut c_void, id: YogStr, title: YogStr) -> bool {
562    let Some(mut env) = get_env() else { return false };
563    let (Some(ji), Some(jt)) = (ys_to_java(&mut env, id), ys_to_java(&mut env, title)) else { return false };
564    env.call_static_method("dev/yog/NativeBridge", "bossbarSetTitle",
565        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&ji), JValue::Object(&jt)])
566    .and_then(|v| v.z()).unwrap_or(false)
567}
568
569unsafe extern "C" fn srv_bossbar_set_progress(_ctx: *mut c_void, id: YogStr, progress: f32) -> bool {
570    let Some(mut env) = get_env() else { return false };
571    let Some(ji) = ys_to_java(&mut env, id) else { return false };
572    env.call_static_method("dev/yog/NativeBridge", "bossbarSetProgress",
573        "(Ljava/lang/String;F)Z", &[JValue::Object(&ji), JValue::Float(progress)])
574    .and_then(|v| v.z()).unwrap_or(false)
575}
576
577unsafe extern "C" fn srv_bossbar_set_color(_ctx: *mut c_void, id: YogStr, color: YogStr) -> bool {
578    let Some(mut env) = get_env() else { return false };
579    let (Some(ji), Some(jc)) = (ys_to_java(&mut env, id), ys_to_java(&mut env, color)) else { return false };
580    env.call_static_method("dev/yog/NativeBridge", "bossbarSetColor",
581        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&ji), JValue::Object(&jc)])
582    .and_then(|v| v.z()).unwrap_or(false)
583}
584
585unsafe extern "C" fn srv_bossbar_add_player(_ctx: *mut c_void, id: YogStr, player: YogStr) -> bool {
586    let Some(mut env) = get_env() else { return false };
587    let (Some(ji), Some(jp)) = (ys_to_java(&mut env, id), ys_to_java(&mut env, player)) else { return false };
588    env.call_static_method("dev/yog/NativeBridge", "bossbarAddPlayer",
589        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&ji), JValue::Object(&jp)])
590    .and_then(|v| v.z()).unwrap_or(false)
591}
592
593unsafe extern "C" fn srv_bossbar_remove_player(_ctx: *mut c_void, id: YogStr, player: YogStr) -> bool {
594    let Some(mut env) = get_env() else { return false };
595    let (Some(ji), Some(jp)) = (ys_to_java(&mut env, id), ys_to_java(&mut env, player)) else { return false };
596    env.call_static_method("dev/yog/NativeBridge", "bossbarRemovePlayer",
597        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&ji), JValue::Object(&jp)])
598    .and_then(|v| v.z()).unwrap_or(false)
599}
600
601unsafe extern "C" fn srv_bossbar_set_visible(_ctx: *mut c_void, id: YogStr, visible: bool) -> bool {
602    let Some(mut env) = get_env() else { return false };
603    let Some(ji) = ys_to_java(&mut env, id) else { return false };
604    env.call_static_method("dev/yog/NativeBridge", "bossbarSetVisible",
605        "(Ljava/lang/String;Z)Z", &[JValue::Object(&ji), JValue::Bool(visible as u8)])
606    .and_then(|v| v.z()).unwrap_or(false)
607}
608
609unsafe extern "C" fn srv_get_block_nbt(_ctx: *mut c_void, dim: YogStr, pos: YogBlockPos) -> YogOwnedStr {
610    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
611    let Some(jd) = ys_to_java(&mut env, dim) else { return YogOwnedStr::NONE };
612    let ret = env.call_static_method("dev/yog/NativeBridge", "getBlockNbt",
613        "(Ljava/lang/String;III)Ljava/lang/String;",
614        &[JValue::Object(&jd), JValue::Int(pos.x), JValue::Int(pos.y), JValue::Int(pos.z)]);
615    match ret.and_then(|v| v.l()) {
616        Ok(obj) => jstring_to_owned(&mut env, obj),
617        _ => YogOwnedStr::NONE,
618    }
619}
620
621unsafe extern "C" fn srv_set_block_nbt(_ctx: *mut c_void, dim: YogStr, pos: YogBlockPos, snbt: YogStr) -> bool {
622    let Some(mut env) = get_env() else { return false };
623    let (Some(jd), Some(js)) = (ys_to_java(&mut env, dim), ys_to_java(&mut env, snbt)) else { return false };
624    env.call_static_method("dev/yog/NativeBridge", "setBlockNbt",
625        "(Ljava/lang/String;IIILjava/lang/String;)Z",
626        &[JValue::Object(&jd), JValue::Int(pos.x), JValue::Int(pos.y), JValue::Int(pos.z), JValue::Object(&js)])
627    .and_then(|v| v.z()).unwrap_or(false)
628}
629
630unsafe extern "C" fn srv_player_inventory(_ctx: *mut c_void, player: YogStr) -> YogOwnedStr {
631    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
632    let Some(jp) = ys_to_java(&mut env, player) else { return YogOwnedStr::NONE };
633    let ret = env.call_static_method("dev/yog/NativeBridge", "playerInventory",
634        "(Ljava/lang/String;)Ljava/lang/String;", &[JValue::Object(&jp)]);
635    match ret.and_then(|v| v.l()) {
636        Ok(obj) => jstring_to_owned(&mut env, obj),
637        _ => YogOwnedStr::NONE,
638    }
639}
640
641unsafe extern "C" fn srv_player_set_slot(_ctx: *mut c_void, player: YogStr, slot: u32, item_id: YogStr, count: u32) -> bool {
642    let Some(mut env) = get_env() else { return false };
643    let (Some(jp), Some(ji)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, item_id)) else { return false };
644    env.call_static_method("dev/yog/NativeBridge", "playerSetSlot",
645        "(Ljava/lang/String;ILjava/lang/String;I)Z",
646        &[JValue::Object(&jp), JValue::Int(slot as i32), JValue::Object(&ji), JValue::Int(count as i32)])
647    .and_then(|v| v.z()).unwrap_or(false)
648}
649
650unsafe extern "C" fn srv_player_teleport_dim(_ctx: *mut c_void, player: YogStr, dim: YogStr, pos: YogVec3) -> bool {
651    let Some(mut env) = get_env() else { return false };
652    let (Some(jp), Some(jd)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, dim)) else { return false };
653    env.call_static_method("dev/yog/NativeBridge", "teleportToDim",
654        "(Ljava/lang/String;Ljava/lang/String;DDD)Z",
655        &[JValue::Object(&jp), JValue::Object(&jd), JValue::Double(pos.x), JValue::Double(pos.y), JValue::Double(pos.z)])
656    .and_then(|v| v.z()).unwrap_or(false)
657}
658
659unsafe extern "C" fn srv_entity_teleport_dim(_ctx: *mut c_void, uuid: YogStr, dim: YogStr, pos: YogVec3) -> bool {
660    let Some(mut env) = get_env() else { return false };
661    let (Some(ju), Some(jd)) = (ys_to_java(&mut env, uuid), ys_to_java(&mut env, dim)) else { return false };
662    env.call_static_method("dev/yog/NativeBridge", "entityTeleportToDim",
663        "(Ljava/lang/String;Ljava/lang/String;DDD)Z",
664        &[JValue::Object(&ju), JValue::Object(&jd), JValue::Double(pos.x), JValue::Double(pos.y), JValue::Double(pos.z)])
665    .and_then(|v| v.z()).unwrap_or(false)
666}
667
668unsafe extern "C" fn srv_online_players(_ctx: *mut c_void) -> YogOwnedStr {
669    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
670    let ret = env.call_static_method("dev/yog/NativeBridge", "onlinePlayers",
671        "()Ljava/lang/String;", &[]);
672    match ret.and_then(|v| v.l()) {
673        Ok(obj) => jstring_to_owned(&mut env, obj),
674        _ => YogOwnedStr::NONE,
675    }
676}
677
678unsafe extern "C" fn srv_world_entity_count(_ctx: *mut c_void, dim: YogStr, entity_type: YogStr) -> i32 {
679    let Some(mut env) = get_env() else { return -1 };
680    let d = dim.as_str();
681    let et = entity_type.as_str();
682    let jd  = match env.new_string(d)  { Ok(s) => s, Err(_) => return -1 };
683    let jet = match env.new_string(et) { Ok(s) => s, Err(_) => return -1 };
684    let ret = env.call_static_method("dev/yog/NativeBridge", "worldEntityCount",
685        "(Ljava/lang/String;Ljava/lang/String;)I",
686        &[JValue::Object(&jd), JValue::Object(&jet)]);
687    match ret.and_then(|v| v.i()) {
688        Ok(n) => n,
689        _ => -1,
690    }
691}
692
693unsafe extern "C" fn srv_game_dir(_ctx: *mut c_void) -> YogOwnedStr {
694    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
695    let ret = env.call_static_method("dev/yog/NativeBridge", "gameDir",
696        "()Ljava/lang/String;", &[]);
697    match ret.and_then(|v| v.l()) {
698        Ok(obj) => jstring_to_owned(&mut env, obj),
699        _ => YogOwnedStr::NONE,
700    }
701}
702
703unsafe extern "C" fn srv_entity_get_nbt(_ctx: *mut c_void, uuid: YogStr) -> YogOwnedStr {
704    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
705    let Some(ju) = ys_to_java(&mut env, uuid) else { return YogOwnedStr::NONE };
706    let ret = env.call_static_method("dev/yog/NativeBridge", "entityGetNbt",
707        "(Ljava/lang/String;)Ljava/lang/String;", &[JValue::Object(&ju)]);
708    match ret.and_then(|v| v.l()) {
709        Ok(obj) => jstring_to_owned(&mut env, obj),
710        _ => YogOwnedStr::NONE,
711    }
712}
713
714unsafe extern "C" fn srv_entity_set_nbt(_ctx: *mut c_void, uuid: YogStr, snbt: YogStr) -> bool {
715    let Some(mut env) = get_env() else { return false };
716    let (Some(ju), Some(js)) = (ys_to_java(&mut env, uuid), ys_to_java(&mut env, snbt)) else { return false };
717    env.call_static_method("dev/yog/NativeBridge", "entitySetNbt",
718        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&ju), JValue::Object(&js)])
719    .and_then(|v| v.z()).unwrap_or(false)
720}
721
722unsafe extern "C" fn srv_spawn_particles(
723    _ctx: *mut c_void, dim: YogStr, pos: YogVec3, particle_type: YogStr,
724    count: i32, dx: f64, dy: f64, dz: f64, speed: f64,
725) -> bool {
726    let Some(mut env) = get_env() else { return false };
727    let (Some(jd), Some(jp)) = (ys_to_java(&mut env, dim), ys_to_java(&mut env, particle_type)) else { return false };
728    env.call_static_method("dev/yog/NativeBridge", "spawnParticles",
729        "(Ljava/lang/String;DDDLjava/lang/String;IDDDD)Z",
730        &[JValue::Object(&jd), JValue::Double(pos.x), JValue::Double(pos.y), JValue::Double(pos.z),
731          JValue::Object(&jp), JValue::Int(count),
732          JValue::Double(dx), JValue::Double(dy), JValue::Double(dz), JValue::Double(speed)])
733    .and_then(|v| v.z()).unwrap_or(false)
734}
735
736unsafe extern "C" fn srv_entity_attribute_get(_ctx: *mut c_void, uuid: YogStr, attr: YogStr) -> f64 {
737    let Some(mut env) = get_env() else { return f64::NAN };
738    let (Some(ju), Some(ja)) = (ys_to_java(&mut env, uuid), ys_to_java(&mut env, attr)) else { return f64::NAN };
739    env.call_static_method("dev/yog/NativeBridge", "entityAttributeGet",
740        "(Ljava/lang/String;Ljava/lang/String;)D", &[JValue::Object(&ju), JValue::Object(&ja)])
741    .and_then(|v| v.d()).unwrap_or(f64::NAN)
742}
743
744unsafe extern "C" fn srv_entity_attribute_set(_ctx: *mut c_void, uuid: YogStr, attr: YogStr, value: f64) -> bool {
745    let Some(mut env) = get_env() else { return false };
746    let (Some(ju), Some(ja)) = (ys_to_java(&mut env, uuid), ys_to_java(&mut env, attr)) else { return false };
747    env.call_static_method("dev/yog/NativeBridge", "entityAttributeSet",
748        "(Ljava/lang/String;Ljava/lang/String;D)Z",
749        &[JValue::Object(&ju), JValue::Object(&ja), JValue::Double(value)])
750    .and_then(|v| v.z()).unwrap_or(false)
751}
752
753unsafe extern "C" fn srv_get_held_item_nbt(_ctx: *mut c_void, player: YogStr) -> YogOwnedStr {
754    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
755    let Some(jp) = ys_to_java(&mut env, player) else { return YogOwnedStr::NONE };
756    let ret = env.call_static_method("dev/yog/NativeBridge", "getHeldItemNbt",
757        "(Ljava/lang/String;)Ljava/lang/String;", &[JValue::Object(&jp)]);
758    match ret.and_then(|v| v.l()) {
759        Ok(obj) => jstring_to_owned(&mut env, obj),
760        _ => YogOwnedStr::NONE,
761    }
762}
763
764unsafe extern "C" fn srv_set_held_item_nbt(_ctx: *mut c_void, player: YogStr, snbt: YogStr) -> bool {
765    let Some(mut env) = get_env() else { return false };
766    let (Some(jp), Some(js)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, snbt)) else { return false };
767    env.call_static_method("dev/yog/NativeBridge", "setHeldItemNbt",
768        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&jp), JValue::Object(&js)])
769    .and_then(|v| v.z()).unwrap_or(false)
770}
771
772unsafe extern "C" fn srv_get_offhand_item_nbt(_ctx: *mut c_void, player: YogStr) -> YogOwnedStr {
773    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
774    let Some(jp) = ys_to_java(&mut env, player) else { return YogOwnedStr::NONE };
775    let ret = env.call_static_method("dev/yog/NativeBridge", "getOffhandItemNbt",
776        "(Ljava/lang/String;)Ljava/lang/String;", &[JValue::Object(&jp)]);
777    match ret.and_then(|v| v.l()) {
778        Ok(obj) => jstring_to_owned(&mut env, obj),
779        _ => YogOwnedStr::NONE,
780    }
781}
782
783unsafe extern "C" fn srv_set_offhand_item_nbt(_ctx: *mut c_void, player: YogStr, snbt: YogStr) -> bool {
784    let Some(mut env) = get_env() else { return false };
785    let (Some(jp), Some(js)) = (ys_to_java(&mut env, player), ys_to_java(&mut env, snbt)) else { return false };
786    env.call_static_method("dev/yog/NativeBridge", "setOffhandItemNbt",
787        "(Ljava/lang/String;Ljava/lang/String;)Z", &[JValue::Object(&jp), JValue::Object(&js)])
788    .and_then(|v| v.z()).unwrap_or(false)
789}
790
791unsafe extern "C" fn srv_get_slot_item(_ctx: *mut c_void, player: YogStr, slot: u32) -> YogOwnedStr {
792    let Some(mut env) = get_env() else { return YogOwnedStr::NONE };
793    let Some(jp) = ys_to_java(&mut env, player) else { return YogOwnedStr::NONE };
794    let ret = env.call_static_method("dev/yog/NativeBridge", "getSlotItem",
795        "(Ljava/lang/String;I)Ljava/lang/String;",
796        &[JValue::Object(&jp), JValue::Int(slot as i32)]);
797    match ret.and_then(|v| v.l()) {
798        Ok(obj) => jstring_to_owned(&mut env, obj),
799        _ => YogOwnedStr::NONE,
800    }
801}
802
803unsafe extern "C" fn srv_set_slot_item(
804    _ctx: *mut c_void, player: YogStr, slot: u32,
805    item_id: YogStr, count: u32, snbt: YogStr,
806) -> bool {
807    let Some(mut env) = get_env() else { return false };
808    let (Some(jp), Some(ji), Some(js)) = (
809        ys_to_java(&mut env, player), ys_to_java(&mut env, item_id), ys_to_java(&mut env, snbt)
810    ) else { return false };
811    env.call_static_method("dev/yog/NativeBridge", "setSlotItem",
812        "(Ljava/lang/String;ILjava/lang/String;ILjava/lang/String;)Z",
813        &[JValue::Object(&jp), JValue::Int(slot as i32), JValue::Object(&ji), JValue::Int(count as i32), JValue::Object(&js)])
814    .and_then(|v| v.z()).unwrap_or(false)
815}
816
817// ── ABI minor 14 — low-level GPU pipeline ────────────────────────────────────
818//
819// All raw GL calls go through `glow::Context` stored in GL.
820// `draw2d_*` functions still call JNI (NativeDraw) for MC text/texture rendering.
821// Everything is called on the render thread — no synchronization needed.
822
823// ── helpers ───────────────────────────────────────────────────────────────────
824
825fn gl_draw_mode(mode: u8) -> u32 {
826    match mode {
827        1 => glow::LINES,
828        2 => glow::LINE_STRIP,
829        3 => glow::TRIANGLE_STRIP,
830        4 => glow::TRIANGLE_FAN,
831        _ => glow::TRIANGLES,
832    }
833}
834
835fn gl_attr_type(dtype: u8) -> u32 {
836    match dtype {
837        1 => glow::UNSIGNED_BYTE,
838        2 => glow::INT,
839        3 => glow::UNSIGNED_INT,
840        _ => glow::FLOAT,
841    }
842}
843
844unsafe fn compile_shader(gl: &glow::Context, stage: u32, src: &str) -> Option<glow::NativeShader> {
845    let sh = gl.create_shader(stage).ok()?;
846    gl.shader_source(sh, src);
847    gl.compile_shader(sh);
848    if !gl.get_shader_compile_status(sh) {
849        yog_logging::error!("yog-gfx shader compile: {}", gl.get_shader_info_log(sh));
850        gl.delete_shader(sh);
851        return None;
852    }
853    Some(sh)
854}
855
856// ── GPU buffers ───────────────────────────────────────────────────────────────
857
858unsafe extern "C" fn gfx_buf_create() -> u32 {
859    let Some(g) = GL.get() else { return 0 };
860    g.0.create_buffer().map(|b| b.0.get()).unwrap_or(0)
861}
862
863unsafe extern "C" fn gfx_buf_delete(handle: u32) {
864    let Some(g) = GL.get() else { return };
865    let Some(n) = NonZeroU32::new(handle) else { return };
866    g.0.delete_buffer(glow::NativeBuffer(n));
867}
868
869unsafe extern "C" fn gfx_buf_data(handle: u32, bytes: *const u8, len: u32, dynamic: bool) {
870    let Some(g) = GL.get() else { return };
871    let Some(n) = NonZeroU32::new(handle) else { return };
872    let gl = &g.0;
873    let data = std::slice::from_raw_parts(bytes, len as usize);
874    let usage = if dynamic { glow::DYNAMIC_DRAW } else { glow::STATIC_DRAW };
875    gl.bind_buffer(glow::ARRAY_BUFFER, Some(glow::NativeBuffer(n)));
876    gl.buffer_data_u8_slice(glow::ARRAY_BUFFER, data, usage);
877    gl.bind_buffer(glow::ARRAY_BUFFER, None);
878}
879
880unsafe extern "C" fn gfx_buf_subdata(handle: u32, offset: u32, bytes: *const u8, len: u32) {
881    let Some(g) = GL.get() else { return };
882    let Some(n) = NonZeroU32::new(handle) else { return };
883    let gl = &g.0;
884    let data = std::slice::from_raw_parts(bytes, len as usize);
885    gl.bind_buffer(glow::ARRAY_BUFFER, Some(glow::NativeBuffer(n)));
886    gl.buffer_sub_data_u8_slice(glow::ARRAY_BUFFER, offset as i32, data);
887    gl.bind_buffer(glow::ARRAY_BUFFER, None);
888}
889
890// ── Vertex arrays ─────────────────────────────────────────────────────────────
891
892unsafe extern "C" fn gfx_vao_create() -> u32 {
893    let Some(g) = GL.get() else { return 0 };
894    g.0.create_vertex_array().map(|v| v.0.get()).unwrap_or(0)
895}
896
897unsafe extern "C" fn gfx_vao_delete(handle: u32) {
898    let Some(g) = GL.get() else { return };
899    let Some(n) = NonZeroU32::new(handle) else { return };
900    g.0.delete_vertex_array(glow::NativeVertexArray(n));
901}
902
903unsafe extern "C" fn gfx_vao_attrib(
904    vao: u32, vbo: u32, index: u32, components: u8,
905    dtype: u8, normalized: bool, stride: u32, offset: u32,
906) {
907    let Some(g) = GL.get() else { return };
908    let (Some(vn), Some(bn)) = (NonZeroU32::new(vao), NonZeroU32::new(vbo)) else { return };
909    let gl = &g.0;
910    gl.bind_vertex_array(Some(glow::NativeVertexArray(vn)));
911    gl.bind_buffer(glow::ARRAY_BUFFER, Some(glow::NativeBuffer(bn)));
912    let gl_type = gl_attr_type(dtype);
913    if dtype == 2 || dtype == 3 {
914        gl.vertex_attrib_pointer_i32(index, components as i32, gl_type, stride as i32, offset as i32);
915    } else {
916        gl.vertex_attrib_pointer_f32(index, components as i32, gl_type, normalized, stride as i32, offset as i32);
917    }
918    gl.enable_vertex_attrib_array(index);
919    gl.bind_vertex_array(None);
920}
921
922unsafe extern "C" fn gfx_vao_set_ebo(vao: u32, ebo: u32) {
923    let Some(g) = GL.get() else { return };
924    let (Some(vn), Some(en)) = (NonZeroU32::new(vao), NonZeroU32::new(ebo)) else { return };
925    let gl = &g.0;
926    gl.bind_vertex_array(Some(glow::NativeVertexArray(vn)));
927    gl.bind_buffer(glow::ELEMENT_ARRAY_BUFFER, Some(glow::NativeBuffer(en)));
928    gl.bind_vertex_array(None);
929}
930
931// ── Shader programs ───────────────────────────────────────────────────────────
932
933/// Returns a path under `~/.cache/yog/shaders/<hash>.ysc` for the given GLSL source pair,
934/// creating the directory if needed.  Returns `None` if the home directory is unknown.
935fn shader_cache_path(vert: &str, frag: &str) -> Option<PathBuf> {
936    use std::hash::{Hash, Hasher};
937    let mut h = std::collections::hash_map::DefaultHasher::new();
938    vert.hash(&mut h);
939    frag.hash(&mut h);
940    let hash = h.finish();
941    let dir = std::env::var("HOME")
942        .map(|home| PathBuf::from(home).join(".cache").join("yog").join("shaders"))
943        .ok()?;
944    std::fs::create_dir_all(&dir).ok()?;
945    Some(dir.join(format!("{hash:016x}.ysc")))
946}
947
948/// Try restoring a program from a binary blob (`[4 LE bytes: GL format][binary…]`).
949/// Returns `Some(handle)` if the driver accepts the binary, `None` otherwise.
950///
951/// Uses `glProgramBinary` (GL 4.1 / ARB_get_program_binary) via the raw pointer
952/// captured in `nativeGlInit`.  Returns `None` if the extension is unavailable.
953unsafe fn load_shader_binary(gl: &glow::Context, data: &[u8]) -> Option<glow::NativeProgram> {
954    if data.len() < 4 { return None; }
955    type ProgramBinaryFn = unsafe extern "system" fn(u32, u32, *const c_void, i32);
956    let fn_ptr = (*GL_PROGRAM_BINARY.get()?)?;
957    let program_binary: ProgramBinaryFn = std::mem::transmute(fn_ptr);
958
959    let fmt = u32::from_le_bytes([data[0], data[1], data[2], data[3]]);
960    let prog = gl.create_program().ok()?;
961    program_binary(prog.0.get(), fmt, data[4..].as_ptr() as *const c_void, (data.len() - 4) as i32);
962    if gl.get_program_link_status(prog) { Some(prog) } else { gl.delete_program(prog); None }
963}
964
965/// Read the compiled binary of a linked program.  Returns empty `Vec` if unsupported.
966unsafe fn get_shader_binary(prog: glow::NativeProgram) -> (Vec<u8>, u32) {
967    type GetProgramivFn       = unsafe extern "system" fn(u32, u32, *mut i32);
968    type GetProgramBinaryFn   = unsafe extern "system" fn(u32, i32, *mut i32, *mut u32, *mut c_void);
969
970    let Some(get_iv_raw)  = GL_GET_PROGRAM_IV.get().and_then(|v| *v) else { return (vec![], 0) };
971    let Some(get_bin_raw) = GL_GET_PROGRAM_BINARY.get().and_then(|v| *v) else { return (vec![], 0) };
972    let get_program_iv:     GetProgramivFn       = std::mem::transmute(get_iv_raw);
973    let get_program_binary: GetProgramBinaryFn   = std::mem::transmute(get_bin_raw);
974
975    // Query binary size via glGetProgramiv(GL_PROGRAM_BINARY_LENGTH).
976    const PROGRAM_BINARY_LENGTH: u32 = 0x8741;
977    let mut size: i32 = 0;
978    get_program_iv(prog.0.get(), PROGRAM_BINARY_LENGTH, &mut size);
979    if size <= 0 { return (vec![], 0); }
980
981    let mut buf = vec![0u8; size as usize];
982    let mut actual_len: i32 = 0;
983    let mut fmt: u32 = 0;
984    get_program_binary(prog.0.get(), size, &mut actual_len, &mut fmt, buf.as_mut_ptr() as *mut c_void);
985    buf.truncate(actual_len.max(0) as usize);
986    (buf, fmt)
987}
988
989unsafe extern "C" fn gfx_prog_create(vert: YogStr, frag: YogStr, out: *mut u32) -> bool {
990    let Some(g) = GL.get() else { return false };
991    let gl = &g.0;
992    let vert_s = vert.as_str();
993    let frag_s = frag.as_str();
994
995    // Fast path: binary shader cache — avoids GLSL re-compilation on subsequent launches.
996    let cache_path = shader_cache_path(vert_s, frag_s);
997    if let Some(ref path) = cache_path {
998        if let Ok(data) = std::fs::read(path) {
999            if let Some(prog) = load_shader_binary(gl, &data) {
1000                *out = prog.0.get();
1001                return true;
1002            }
1003            // Cache stale (driver updated?); remove and fall through to GLSL compile.
1004            let _ = std::fs::remove_file(path);
1005        }
1006    }
1007
1008    // GLSL compile path.
1009    let vs = match compile_shader(gl, glow::VERTEX_SHADER, vert_s) {
1010        Some(s) => s,
1011        None => return false,
1012    };
1013    let fs = match compile_shader(gl, glow::FRAGMENT_SHADER, frag_s) {
1014        Some(s) => s,
1015        None => { gl.delete_shader(vs); return false; }
1016    };
1017    let prog = match gl.create_program() {
1018        Ok(p) => p,
1019        Err(e) => {
1020            yog_logging::error!("yog-gfx: create_program: {}", e);
1021            gl.delete_shader(vs); gl.delete_shader(fs);
1022            return false;
1023        }
1024    };
1025    gl.attach_shader(prog, vs);
1026    gl.attach_shader(prog, fs);
1027    gl.link_program(prog);
1028    gl.detach_shader(prog, vs);
1029    gl.detach_shader(prog, fs);
1030    gl.delete_shader(vs);
1031    gl.delete_shader(fs);
1032    if !gl.get_program_link_status(prog) {
1033        yog_logging::error!("yog-gfx: shader link: {}", gl.get_program_info_log(prog));
1034        gl.delete_program(prog);
1035        return false;
1036    }
1037
1038    // Persist binary so the next launch skips GLSL compilation entirely.
1039    if let Some(ref path) = cache_path {
1040        let (binary, fmt) = get_shader_binary(prog);
1041        if !binary.is_empty() {
1042            let mut blob = fmt.to_le_bytes().to_vec();
1043            blob.extend_from_slice(&binary);
1044            let _ = std::fs::write(path, &blob);
1045        }
1046    }
1047
1048    *out = prog.0.get();
1049    true
1050}
1051
1052unsafe extern "C" fn gfx_prog_delete(handle: u32) {
1053    let Some(g) = GL.get() else { return };
1054    let Some(n) = NonZeroU32::new(handle) else { return };
1055    g.0.delete_program(glow::NativeProgram(n));
1056}
1057
1058macro_rules! with_prog {
1059    ($handle:expr, |$gl:ident, $prog:ident, $loc:ident ($name:expr)| $body:expr) => {{
1060        let Some(g) = GL.get() else { return };
1061        let Some(n) = NonZeroU32::new($handle) else { return };
1062        let $gl = &g.0;
1063        let $prog = glow::NativeProgram(n);
1064        $gl.use_program(Some($prog));
1065        let $loc = $gl.get_uniform_location($prog, $name.as_str());
1066        $body
1067    }};
1068}
1069
1070unsafe extern "C" fn gfx_prog_uniform_1i(prog: u32, name: YogStr, v: i32) {
1071    with_prog!(prog, |gl, _p, loc(name)| gl.uniform_1_i32(loc.as_ref(), v));
1072}
1073unsafe extern "C" fn gfx_prog_uniform_1f(prog: u32, name: YogStr, v: f32) {
1074    with_prog!(prog, |gl, _p, loc(name)| gl.uniform_1_f32(loc.as_ref(), v));
1075}
1076unsafe extern "C" fn gfx_prog_uniform_2f(prog: u32, name: YogStr, x: f32, y: f32) {
1077    with_prog!(prog, |gl, _p, loc(name)| gl.uniform_2_f32(loc.as_ref(), x, y));
1078}
1079unsafe extern "C" fn gfx_prog_uniform_3f(prog: u32, name: YogStr, x: f32, y: f32, z: f32) {
1080    with_prog!(prog, |gl, _p, loc(name)| gl.uniform_3_f32(loc.as_ref(), x, y, z));
1081}
1082unsafe extern "C" fn gfx_prog_uniform_4f(prog: u32, name: YogStr, x: f32, y: f32, z: f32, w: f32) {
1083    with_prog!(prog, |gl, _p, loc(name)| gl.uniform_4_f32(loc.as_ref(), x, y, z, w));
1084}
1085unsafe extern "C" fn gfx_prog_uniform_mat4(prog: u32, name: YogStr, col_major: *const f32) {
1086    with_prog!(prog, |gl, _p, loc(name)| {
1087        let data = std::slice::from_raw_parts(col_major, 16);
1088        gl.uniform_matrix_4_f32_slice(loc.as_ref(), false, data);
1089    });
1090}
1091
1092// ── Textures ──────────────────────────────────────────────────────────────────
1093
1094unsafe extern "C" fn gfx_tex_create(w: u32, h: u32, rgba: *const u8, linear: bool) -> u32 {
1095    let Some(g) = GL.get() else { return 0 };
1096    let gl = &g.0;
1097    let tex = match gl.create_texture() { Ok(t) => t, Err(_) => return 0 };
1098    gl.bind_texture(glow::TEXTURE_2D, Some(tex));
1099    let filter = if linear { glow::LINEAR } else { glow::NEAREST };
1100    gl.tex_parameter_i32(glow::TEXTURE_2D, glow::TEXTURE_MIN_FILTER, filter as i32);
1101    gl.tex_parameter_i32(glow::TEXTURE_2D, glow::TEXTURE_MAG_FILTER, filter as i32);
1102    gl.tex_parameter_i32(glow::TEXTURE_2D, glow::TEXTURE_WRAP_S, glow::CLAMP_TO_EDGE as i32);
1103    gl.tex_parameter_i32(glow::TEXTURE_2D, glow::TEXTURE_WRAP_T, glow::CLAMP_TO_EDGE as i32);
1104    let pixels = std::slice::from_raw_parts(rgba, (w * h * 4) as usize);
1105    gl.tex_image_2d(
1106        glow::TEXTURE_2D, 0, glow::RGBA8 as i32,
1107        w as i32, h as i32, 0,
1108        glow::RGBA, glow::UNSIGNED_BYTE,
1109        Some(pixels),
1110    );
1111    gl.bind_texture(glow::TEXTURE_2D, None);
1112    tex.0.get()
1113}
1114
1115unsafe extern "C" fn gfx_tex_delete(handle: u32) {
1116    let Some(g) = GL.get() else { return };
1117    let Some(n) = NonZeroU32::new(handle) else { return };
1118    g.0.delete_texture(glow::NativeTexture(n));
1119}
1120
1121unsafe extern "C" fn gfx_tex_bind(unit: u32, handle: u32) {
1122    let Some(g) = GL.get() else { return };
1123    let gl = &g.0;
1124    gl.active_texture(glow::TEXTURE0 + unit);
1125    match NonZeroU32::new(handle) {
1126        Some(n) => gl.bind_texture(glow::TEXTURE_2D, Some(glow::NativeTexture(n))),
1127        None    => gl.bind_texture(glow::TEXTURE_2D, None),
1128    }
1129}
1130
1131unsafe extern "C" fn gfx_tex_from_mc(id: YogStr) -> u32 {
1132    let Some(mut env) = get_env() else { return 0 };
1133    let Some(ji) = ys_to_java(&mut env, id) else { return 0 };
1134    env.call_static_method("dev/yog/NativeDraw", "getMcTextureId",
1135        "(Ljava/lang/String;)I", &[JValue::Object(&ji)])
1136        .and_then(|v| v.i())
1137        .map(|id| id as u32)
1138        .unwrap_or(0)
1139}
1140
1141// ── Draw calls ────────────────────────────────────────────────────────────────
1142
1143unsafe extern "C" fn gfx_draw_arrays(vao: u32, prog: u32, mode: u8, first: u32, count: u32) {
1144    let Some(g) = GL.get() else { return };
1145    let (Some(vn), Some(pn)) = (NonZeroU32::new(vao), NonZeroU32::new(prog)) else { return };
1146    let gl = &g.0;
1147    gl.use_program(Some(glow::NativeProgram(pn)));
1148    gl.bind_vertex_array(Some(glow::NativeVertexArray(vn)));
1149    gl.draw_arrays(gl_draw_mode(mode), first as i32, count as i32);
1150    gl.bind_vertex_array(None);
1151}
1152
1153unsafe extern "C" fn gfx_draw_elements(vao: u32, ebo: u32, prog: u32, mode: u8, count: u32, u32_idx: bool) {
1154    let Some(g) = GL.get() else { return };
1155    let (Some(vn), Some(pn)) = (NonZeroU32::new(vao), NonZeroU32::new(prog)) else { return };
1156    let gl = &g.0;
1157    gl.use_program(Some(glow::NativeProgram(pn)));
1158    gl.bind_vertex_array(Some(glow::NativeVertexArray(vn)));
1159    // EBO is stored in the VAO; ebo param is informational for safety but not re-bound here.
1160    let _ = ebo;
1161    let idx_type = if u32_idx { glow::UNSIGNED_INT } else { glow::UNSIGNED_SHORT };
1162    gl.draw_elements(gl_draw_mode(mode), count as i32, idx_type, 0);
1163    gl.bind_vertex_array(None);
1164}
1165
1166// ── Render state ──────────────────────────────────────────────────────────────
1167
1168unsafe extern "C" fn gfx_set_blend(enabled: bool, src: u32, dst: u32) {
1169    let Some(g) = GL.get() else { return };
1170    let gl = &g.0;
1171    if enabled {
1172        gl.enable(glow::BLEND);
1173        gl.blend_func(src, dst);
1174    } else {
1175        gl.disable(glow::BLEND);
1176    }
1177}
1178
1179unsafe extern "C" fn gfx_set_depth(test: bool, write: bool) {
1180    let Some(g) = GL.get() else { return };
1181    let gl = &g.0;
1182    if test { gl.enable(glow::DEPTH_TEST); } else { gl.disable(glow::DEPTH_TEST); }
1183    gl.depth_mask(write);
1184}
1185
1186unsafe extern "C" fn gfx_set_scissor(x: i32, y: i32, w: i32, h: i32) {
1187    let Some(g) = GL.get() else { return };
1188    let gl = &g.0;
1189    gl.enable(glow::SCISSOR_TEST);
1190    gl.scissor(x, y, w, h);
1191}
1192
1193unsafe extern "C" fn gfx_clear_scissor() {
1194    let Some(g) = GL.get() else { return };
1195    g.0.disable(glow::SCISSOR_TEST);
1196}
1197
1198unsafe extern "C" fn gfx_set_viewport(x: i32, y: i32, w: i32, h: i32) {
1199    let Some(g) = GL.get() else { return };
1200    g.0.viewport(x, y, w, h);
1201}
1202
1203// ── 2D convenience (JNI — uses MC's DrawContext / text renderer) ──────────────
1204
1205unsafe extern "C" fn gfx_draw2d_rect(x1: f32, y1: f32, x2: f32, y2: f32, color: u32) {
1206    let Some(mut env) = get_env() else { return };
1207    let _ = env.call_static_method("dev/yog/NativeDraw", "drawRect",
1208        "(FFFFI)V",
1209        &[JValue::Float(x1), JValue::Float(y1), JValue::Float(x2), JValue::Float(y2),
1210          JValue::Int(color as i32)]);
1211}
1212
1213unsafe extern "C" fn gfx_draw2d_gradient(x1: f32, y1: f32, x2: f32, y2: f32, top: u32, bottom: u32) {
1214    let Some(mut env) = get_env() else { return };
1215    let _ = env.call_static_method("dev/yog/NativeDraw", "drawGradientRect",
1216        "(FFFFII)V",
1217        &[JValue::Float(x1), JValue::Float(y1), JValue::Float(x2), JValue::Float(y2),
1218          JValue::Int(top as i32), JValue::Int(bottom as i32)]);
1219}
1220
1221unsafe extern "C" fn gfx_draw2d_text(text: YogStr, x: f32, y: f32, color: u32, shadow: bool) {
1222    let Some(mut env) = get_env() else { return };
1223    if let Some(jt) = ys_to_java(&mut env, text) {
1224        let _ = env.call_static_method("dev/yog/NativeDraw", "drawText",
1225            "(Ljava/lang/String;FFIZ)V",
1226            &[JValue::Object(&jt), JValue::Float(x), JValue::Float(y),
1227              JValue::Int(color as i32), JValue::Bool(shadow as u8)]);
1228    }
1229}
1230
1231unsafe extern "C" fn gfx_draw2d_mc_tex(
1232    id: YogStr, x: f32, y: f32, u0: f32, v0: f32, w: f32, h: f32, tw: f32, th: f32,
1233) {
1234    let Some(mut env) = get_env() else { return };
1235    if let Some(ji) = ys_to_java(&mut env, id) {
1236        let _ = env.call_static_method("dev/yog/NativeDraw", "drawTexture",
1237            "(Ljava/lang/String;FFFFFFFFF)V",
1238            &[JValue::Object(&ji),
1239              JValue::Float(x), JValue::Float(y), JValue::Float(u0), JValue::Float(v0),
1240              JValue::Float(w), JValue::Float(h), JValue::Float(tw), JValue::Float(th)]);
1241    }
1242}
1243
1244// ── Static GFX function table (function pointers only — per-frame data filled at call time) ──
1245
1246static GFX_FN_TABLE: YogGfxApi = YogGfxApi {
1247    // Per-frame fields zeroed in the static; actual values are set on the stack per render call.
1248    screen_w: 0, screen_h: 0, delta_tick: 0.0, scale_factor: 1.0,
1249    view_proj: [0.0; 16], camera_pos: [0.0; 3], player_pos: [0.0; 3], _pad1: 0.0,
1250    buf_create:        gfx_buf_create,
1251    buf_delete:        gfx_buf_delete,
1252    buf_data:          gfx_buf_data,
1253    buf_subdata:       gfx_buf_subdata,
1254    vao_create:        gfx_vao_create,
1255    vao_delete:        gfx_vao_delete,
1256    vao_attrib:        gfx_vao_attrib,
1257    vao_set_ebo:       gfx_vao_set_ebo,
1258    prog_create:       gfx_prog_create,
1259    prog_delete:       gfx_prog_delete,
1260    prog_uniform_1i:   gfx_prog_uniform_1i,
1261    prog_uniform_1f:   gfx_prog_uniform_1f,
1262    prog_uniform_2f:   gfx_prog_uniform_2f,
1263    prog_uniform_3f:   gfx_prog_uniform_3f,
1264    prog_uniform_4f:   gfx_prog_uniform_4f,
1265    prog_uniform_mat4: gfx_prog_uniform_mat4,
1266    tex_create:        gfx_tex_create,
1267    tex_delete:        gfx_tex_delete,
1268    tex_bind:          gfx_tex_bind,
1269    tex_from_mc:       gfx_tex_from_mc,
1270    draw_arrays:       gfx_draw_arrays,
1271    draw_elements:     gfx_draw_elements,
1272    set_blend:         gfx_set_blend,
1273    set_depth:         gfx_set_depth,
1274    set_scissor:       gfx_set_scissor,
1275    clear_scissor:     gfx_clear_scissor,
1276    set_viewport:      gfx_set_viewport,
1277    draw2d_rect:       gfx_draw2d_rect,
1278    draw2d_gradient:   gfx_draw2d_gradient,
1279    draw2d_text:       gfx_draw2d_text,
1280    draw2d_mc_tex:     gfx_draw2d_mc_tex,
1281};
1282
1283// ── YogApi registration functions ─────────────────────────────────────────────
1284//
1285// ctx is *mut RuntimeHandlers (cast from *mut c_void).
1286
1287macro_rules! api_event {
1288    ($name:ident, $field:ident, $fn_ty:ty) => {
1289        unsafe extern "C" fn $name(ctx: *mut c_void, ud: *mut c_void, h: $fn_ty) {
1290            let handlers = &mut *(ctx as *mut RuntimeHandlers);
1291            handlers.$field.push((ud, h));
1292        }
1293    };
1294}
1295
1296api_event!(api_on_block_break,      block_break,        YogBlockBreakFn);
1297api_event!(api_on_chat,             chat,               YogChatFn);
1298api_event!(api_on_player_join,      player_join,        YogPlayerFn);
1299api_event!(api_on_player_leave,     player_leave,       YogPlayerFn);
1300api_event!(api_on_use_item,         use_item,           YogUseItemFn);
1301api_event!(api_on_use_block,        use_block,          YogUseBlockFn);
1302api_event!(api_on_attack_entity,    attack_entity,      YogAttackEntityFn);
1303api_event!(api_on_entity_damage,    entity_damage,      YogEntityDamageFn);
1304api_event!(api_on_entity_death,     entity_death,       YogEntityDeathFn);
1305api_event!(api_on_entity_spawn,     entity_spawn,       YogEntitySpawnFn);
1306api_event!(api_on_player_place_block, player_place_block, YogPlaceBlockFn);
1307api_event!(api_on_player_death,     player_death,       YogPlayerDeathFn);
1308api_event!(api_on_player_respawn,   player_respawn,     YogPlayerRespawnFn);
1309api_event!(api_on_advancement,      advancement,        YogAdvancementFn);
1310api_event!(api_on_entity_interact,  entity_interact,    YogEntityInteractFn);
1311api_event!(api_on_item_craft,       item_craft,         YogCraftFn);
1312api_event!(api_on_explosion,        explosion,          YogExplosionFn);
1313api_event!(api_on_item_pickup,      item_pickup,        YogItemPickupFn);
1314api_event!(api_on_player_move,      player_move,        YogPlayerMoveFn);
1315api_event!(api_on_container_open,   container_open,     YogContainerOpenFn);
1316api_event!(api_on_container_close,  container_close,    YogContainerCloseFn);
1317api_event!(api_on_projectile_hit,   projectile_hit,     YogProjectileHitFn);
1318api_event!(api_on_client_tick,      client_tick,        YogClientFn);
1319api_event!(api_on_hud_render,       hud_render,         YogHudRenderFn);
1320api_event!(api_on_world_render,     world_render,       YogWorldRenderFn);
1321api_event!(api_on_key_press,        key_press,          YogKeyPressFn);
1322api_event!(api_on_screen_open,      screen_open,        YogScreenFn);
1323api_event!(api_on_screen_close,     screen_close,       YogScreenFn);
1324api_event!(api_on_server_tick,      server_tick,        YogServerFn);
1325api_event!(api_on_server_started,   server_started,     YogServerFn);
1326api_event!(api_on_server_stopping,  server_stopping,    YogServerFn);
1327
1328unsafe extern "C" fn api_on_packet(ctx: *mut c_void, channel: YogStr, ud: *mut c_void, h: YogPacketFn) {
1329    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1330    handlers.packets.insert(channel.as_str().to_owned(), (ud, h));
1331}
1332
1333unsafe extern "C" fn api_on_client_packet(ctx: *mut c_void, channel: YogStr, ud: *mut c_void, h: YogPacketFn) {
1334    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1335    handlers.client_packets.insert(channel.as_str().to_owned(), (ud, h));
1336}
1337
1338unsafe extern "C" fn api_register_command(ctx: *mut c_void, name: YogStr, ud: *mut c_void, h: YogCommandFn) {
1339    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1340    handlers.commands.insert(name.as_str().to_owned(), (ud, h));
1341}
1342
1343unsafe extern "C" fn api_register_typed_command(ctx: *mut c_void, name: YogStr, schema: YogStr, ud: *mut c_void, h: YogCommandFn) {
1344    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1345    let n = name.as_str().to_owned();
1346    handlers.typed_schemas.insert(n.clone(), schema.as_str().to_owned());
1347    handlers.commands.insert(n, (ud, h));
1348}
1349
1350
1351unsafe extern "C" fn api_register_recipe_json(ctx: *mut c_void, namespace: YogStr, name: YogStr, json: YogStr) {
1352    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1353    handlers.recipes.push((
1354        namespace.as_str().to_owned(),
1355        name.as_str().to_owned(),
1356        json.as_str().to_owned(),
1357    ));
1358}
1359
1360unsafe extern "C" fn api_register_item(ctx: *mut c_void, def: *const YogItemDef) {
1361    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1362    let d = &*def;
1363    let food = if d.food_nutrition > 0 {
1364        Some(FoodDef { nutrition: d.food_nutrition, saturation: d.food_saturation, can_always_eat: d.food_always_eat })
1365    } else { None };
1366    handlers.items.push(ItemDef {
1367        id:            d.id.as_str().to_owned(),
1368        max_stack:     d.max_stack as u8,
1369        name:          if d.name.is_empty() { None } else { Some(d.name.as_str().to_owned()) },
1370        tooltip:       if d.tooltip.is_empty() { None } else { Some(d.tooltip.as_str().to_owned()) },
1371        max_damage:    d.max_damage,
1372        fire_resistant: d.fire_resistant,
1373        fuel_ticks:    d.fuel_ticks,
1374        food,
1375    });
1376}
1377
1378unsafe extern "C" fn api_register_block(ctx: *mut c_void, def: *const YogBlockDef) {
1379    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1380    let d = &*def;
1381    handlers.blocks.push(BlockDef {
1382        id:            d.id.as_str().to_owned(),
1383        hardness:      d.hardness,
1384        resistance:    d.resistance,
1385        name:          if d.name.is_empty() { None } else { Some(d.name.as_str().to_owned()) },
1386        light_level:   d.light_level,
1387        sound:         if d.sound.is_empty() { None } else { Some(d.sound.as_str().to_owned()) },
1388        requires_tool: d.requires_tool,
1389        no_collision:  d.no_collision,
1390        slipperiness:  d.slipperiness,
1391        shape:         if d.shape == [0.0f32; 6] { None } else { Some(d.shape) },
1392    });
1393}
1394
1395unsafe extern "C" fn api_schedule_once(ctx: *mut c_void, delay_ticks: u64, ud: *mut c_void, h: YogScheduledFn) {
1396    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1397    handlers.scheduler.lock().expect("scheduler poisoned").once_tasks.push(OnceTask { delay_remaining: delay_ticks, ud, f: h });
1398}
1399
1400unsafe extern "C" fn api_schedule_repeating(ctx: *mut c_void, period_ticks: u64, ud: *mut c_void, h: YogScheduledFn) {
1401    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1402    handlers.scheduler.lock().expect("scheduler poisoned").repeating_tasks.push(RepeatingTask { period: period_ticks, ticks_left: period_ticks, ud, f: h });
1403}
1404
1405unsafe extern "C" fn api_register_startup_grant(ctx: *mut c_void, grant: *const YogStartupGrantDef) {
1406    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1407    let g = &*grant;
1408    let items: Vec<String> = if g.items.is_empty() {
1409        Vec::new()
1410    } else {
1411        unsafe { g.items.as_str().split('|').map(|s: &str| s.to_owned()).collect() }
1412    };
1413    let book = if g.book.is_empty() { None } else { Some(unsafe { g.book.as_str().to_owned() }) };
1414    let command = if g.command.is_empty() { None } else { Some(unsafe { g.command.as_str().to_owned() }) };
1415    handlers.startup_grants.push(yog_registry::StartupGrant {
1416        id: unsafe { g.id.as_str().to_owned() },
1417        items,
1418        book,
1419        command,
1420    });
1421}
1422
1423unsafe extern "C" fn api_register_book(ctx: *mut c_void, book_id: YogStr, book_json: YogStr) {
1424    let handlers = &mut *(ctx as *mut RuntimeHandlers);
1425    handlers.books.insert(unsafe { book_id.as_str().to_owned() }, unsafe { book_json.as_str().to_owned() });
1426}
1427
1428#[no_mangle]
1429pub extern "system" fn Java_dev_yog_NativeBridge_nativeBookJson<'l>(
1430    mut env: JNIEnv<'l>, _class: JClass<'l>, book_id: JString<'l>,
1431) -> jstring {
1432    let id = match env.get_string(&book_id) {
1433        Ok(s) => String::from(s),
1434        Err(_) => return std::ptr::null_mut(),
1435    };
1436    let json = handlers().books.get(&id).cloned().unwrap_or_else(|| "null".to_string());
1437    env.new_string(json).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
1438}
1439
1440// ── Table constructors ────────────────────────────────────────────────────────
1441
1442fn build_server_table() -> YogServer {
1443    YogServer {
1444        ctx:         std::ptr::null_mut(),
1445        abi_version: ABI_VERSION,
1446        size:        std::mem::size_of::<YogServer>() as u32,
1447        free_str:    yog_free_str,
1448        broadcast:   srv_broadcast,
1449        get_block:   srv_get_block,
1450        set_block:   srv_set_block,
1451        world_time:  srv_world_time,
1452        set_time:    srv_set_time,
1453        is_raining:  srv_is_raining,
1454        set_weather: srv_set_weather,
1455        give_item:   srv_give_item,
1456        player_teleport: srv_player_teleport,
1457        send_to_player: srv_send_to_player,
1458        send_to_server: srv_send_to_server,
1459        kick_player: srv_kick_player,
1460        set_gamemode: srv_set_gamemode,
1461        send_title:  srv_send_title,
1462        send_actionbar: srv_send_actionbar,
1463        play_sound:  srv_play_sound,
1464        play_sound_player: srv_play_sound_player,
1465        entity_teleport: srv_entity_teleport,
1466        entity_position: srv_entity_position,
1467        entity_health: srv_entity_health,
1468        entity_set_health: srv_entity_set_health,
1469        entity_kill: srv_entity_kill,
1470        spawn_entity: srv_spawn_entity,
1471        entity_add_effect: srv_entity_add_effect,
1472        entity_remove_effect: srv_entity_remove_effect,
1473        entity_clear_effects: srv_entity_clear_effects,
1474        entity_velocity: srv_entity_velocity,
1475        entity_set_velocity: srv_entity_set_velocity,
1476        entity_add_velocity: srv_entity_add_velocity,
1477        has_item_tag: srv_has_item_tag,
1478        has_block_tag: srv_has_block_tag,
1479        drop_loot: srv_drop_loot,
1480        scoreboard_get: srv_scoreboard_get,
1481        scoreboard_set: srv_scoreboard_set,
1482        scoreboard_add: srv_scoreboard_add,
1483        bossbar_create: srv_bossbar_create,
1484        bossbar_remove: srv_bossbar_remove,
1485        bossbar_set_title: srv_bossbar_set_title,
1486        bossbar_set_progress: srv_bossbar_set_progress,
1487        bossbar_set_color: srv_bossbar_set_color,
1488        bossbar_add_player: srv_bossbar_add_player,
1489        bossbar_remove_player: srv_bossbar_remove_player,
1490        bossbar_set_visible: srv_bossbar_set_visible,
1491        game_dir: srv_game_dir,
1492        get_block_nbt:       srv_get_block_nbt,
1493        set_block_nbt:       srv_set_block_nbt,
1494        player_inventory:    srv_player_inventory,
1495        player_set_slot:     srv_player_set_slot,
1496        player_teleport_dim: srv_player_teleport_dim,
1497        entity_teleport_dim: srv_entity_teleport_dim,
1498        online_players:      srv_online_players,
1499        world_entity_count:  srv_world_entity_count,
1500        entity_get_nbt:          srv_entity_get_nbt,
1501        entity_set_nbt:          srv_entity_set_nbt,
1502        spawn_particles:         srv_spawn_particles,
1503        entity_attribute_get:    srv_entity_attribute_get,
1504        entity_attribute_set:    srv_entity_attribute_set,
1505        get_held_item_nbt:       srv_get_held_item_nbt,
1506        set_held_item_nbt:       srv_set_held_item_nbt,
1507        get_offhand_item_nbt:    srv_get_offhand_item_nbt,
1508        set_offhand_item_nbt:    srv_set_offhand_item_nbt,
1509        get_slot_item:           srv_get_slot_item,
1510        set_slot_item:           srv_set_slot_item,
1511    }
1512}
1513
1514fn build_api_table(ctx: *mut RuntimeHandlers, server: *const YogServer) -> YogApi {
1515    YogApi {
1516        abi_version: ABI_VERSION,
1517        size:        std::mem::size_of::<YogApi>() as u32,
1518        ctx:         ctx as *mut c_void,
1519        server,
1520        on_block_break:     api_on_block_break,
1521        on_chat:            api_on_chat,
1522        on_player_join:     api_on_player_join,
1523        on_player_leave:    api_on_player_leave,
1524        on_use_item:        api_on_use_item,
1525        on_use_block:       api_on_use_block,
1526        on_attack_entity:   api_on_attack_entity,
1527        on_entity_damage:   api_on_entity_damage,
1528        on_entity_death:    api_on_entity_death,
1529        on_entity_spawn:         api_on_entity_spawn,
1530        on_player_place_block:   api_on_player_place_block,
1531        on_player_death:         api_on_player_death,
1532        on_player_respawn:       api_on_player_respawn,
1533        on_advancement:          api_on_advancement,
1534        on_entity_interact:      api_on_entity_interact,
1535        on_item_craft:           api_on_item_craft,
1536        on_explosion:            api_on_explosion,
1537        on_item_pickup:          api_on_item_pickup,
1538        on_player_move:          api_on_player_move,
1539        on_container_open:       api_on_container_open,
1540        on_container_close:      api_on_container_close,
1541        on_projectile_hit:       api_on_projectile_hit,
1542        on_server_tick:          api_on_server_tick,
1543        on_server_started:       api_on_server_started,
1544        on_server_stopping:      api_on_server_stopping,
1545        on_packet:               api_on_packet,
1546        on_client_packet:        api_on_client_packet,
1547        register_command:        api_register_command,
1548        register_typed_command:  api_register_typed_command,
1549        register_recipe_json:    api_register_recipe_json,
1550        register_item:          api_register_item,
1551        register_block:     api_register_block,
1552        schedule_once:      api_schedule_once,
1553        schedule_repeating: api_schedule_repeating,
1554        on_client_tick:     api_on_client_tick,
1555        on_hud_render:      api_on_hud_render,
1556        on_key_press:       api_on_key_press,
1557        on_screen_open:     api_on_screen_open,
1558        on_screen_close:    api_on_screen_close,
1559        on_world_render:    api_on_world_render,
1560        register_startup_grant: api_register_startup_grant,
1561        register_book:          api_register_book,
1562    }
1563}
1564
1565// ── Mod loading ───────────────────────────────────────────────────────────────
1566
1567fn platform_tag() -> String {
1568    format!("{}-{}", std::env::consts::OS, std::env::consts::ARCH)
1569}
1570
1571type AbiVersionFn   = unsafe extern "C" fn() -> u32;
1572type RegisterFn     = unsafe extern "C" fn(*const YogApi, *mut c_void);
1573
1574fn load_mods(dir: &Path, api: &YogApi) {
1575    let entries = match std::fs::read_dir(dir) {
1576        Ok(e) => e,
1577        Err(_) => {
1578            yog_logging::info!("no mods directory at {} — none loaded", dir.display());
1579            return;
1580        }
1581    };
1582    let mut count = 0u32;
1583    for entry in entries.flatten() {
1584        let path = entry.path();
1585        let lib_path = match path.extension().and_then(|e| e.to_str()) {
1586            Some("yog") => match extract_yog(&path) {
1587                Some(p) => p,
1588                None => {
1589                    yog_logging::error!("no native for {} in {}", platform_tag(), path.display());
1590                    continue;
1591                }
1592            },
1593            Some("so") | Some("dll") | Some("dylib") => path.clone(),
1594            _ => continue,
1595        };
1596        if load_mod_lib(&lib_path, api) { count += 1; }
1597    }
1598    yog_logging::info!("loaded {} mod(s) from {}", count, dir.display());
1599}
1600
1601fn load_mod_lib(path: &Path, api: &YogApi) -> bool {
1602    unsafe {
1603        let lib = match Library::new(path) {
1604            Ok(l) => l,
1605            Err(e) => { yog_logging::error!("failed to load {}: {}", path.display(), e); return false; }
1606        };
1607        let abi: Symbol<AbiVersionFn> = match lib.get(b"yog_abi_version") {
1608            Ok(s) => s,
1609            Err(_) => { yog_logging::error!("{} is not a Yog mod (no yog_abi_version)", path.display()); return false; }
1610        };
1611        let mod_abi = abi();
1612        if mod_abi != ABI_VERSION {
1613            yog_logging::error!("{}: ABI {} incompatible with runtime ABI {}", path.display(), mod_abi, ABI_VERSION);
1614            return false;
1615        }
1616        let register: Symbol<RegisterFn> = match lib.get(b"yog_mod_register") {
1617            Ok(s) => s,
1618            Err(_) => { yog_logging::error!("{} missing yog_mod_register", path.display()); return false; }
1619        };
1620        register(api as *const YogApi, std::ptr::null_mut());
1621        drop(register);
1622        drop(abi);
1623        LOADED_MODS.lock().expect("mods lock poisoned").push(lib);
1624    }
1625    true
1626}
1627
1628fn extract_yog(path: &Path) -> Option<PathBuf> {
1629    let file = std::fs::File::open(path).ok()?;
1630    let mut archive = zip::ZipArchive::new(file).ok()?;
1631    let prefix = format!("natives/{}/", platform_tag());
1632    let mut entry_name = None;
1633    for i in 0..archive.len() {
1634        let f = archive.by_index(i).ok()?;
1635        if f.name().starts_with(&prefix) && !f.name().ends_with('/') {
1636            entry_name = Some(f.name().to_string());
1637            break;
1638        }
1639    }
1640    let entry_name = entry_name?;
1641    let ext = Path::new(&entry_name).extension().and_then(|e| e.to_str()).unwrap_or("bin");
1642    let stem = path.file_stem()?.to_string_lossy().into_owned();
1643    let out = std::env::temp_dir().join(format!("yog-{}-{}.{}", stem, std::process::id(), ext));
1644    let mut entry = archive.by_name(&entry_name).ok()?;
1645    let mut out_file = std::fs::File::create(&out).ok()?;
1646    std::io::copy(&mut entry, &mut out_file).ok()?;
1647    Some(out)
1648}
1649
1650// ── Dispatcher helpers ────────────────────────────────────────────────────────
1651
1652fn srv_ptr() -> *const YogServer {
1653    SERVER.get().expect("yog: SERVER not initialised") as *const YogServer
1654}
1655
1656// ── JNI entry points ──────────────────────────────────────────────────────────
1657
1658#[no_mangle]
1659pub extern "system" fn Java_dev_yog_NativeBridge_nativeInit<'l>(
1660    mut env: JNIEnv<'l>,
1661    _class: JClass<'l>,
1662    mods_dir: JString<'l>,
1663) {
1664    if let Ok(vm) = env.get_java_vm() { let _ = JAVA_VM.set(vm); }
1665
1666    let dir = env.get_string(&mods_dir).map(String::from).unwrap_or_default();
1667
1668    // Build YogServer and store in static (gets a stable address).
1669    let _ = SERVER.set(build_server_table());
1670    let server_ptr = SERVER.get().unwrap() as *const YogServer;
1671
1672    // Build RuntimeHandlers on the heap temporarily so we have a stable pointer
1673    // to pass as ctx while mods register.
1674    let mut handlers = Box::new(RuntimeHandlers::new());
1675    let handlers_ptr = &mut *handlers as *mut RuntimeHandlers;
1676
1677    let api = build_api_table(handlers_ptr, server_ptr);
1678
1679    guard("mod loading", || {
1680        load_mods(Path::new(&dir), &api);
1681    });
1682
1683    // Move handlers out of Box and into the OnceLock.
1684    let _ = HANDLERS.set(*handlers);
1685
1686    yog_logging::info!("runtime initialised — the gate is open.");
1687}
1688
1689#[no_mangle]
1690pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnBlockBreak<'l>(
1691    mut env: JNIEnv<'l>, _class: JClass<'l>,
1692    player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
1693) {
1694    let (p, b) = (jstr!(env, player), jstr!(env, block));
1695    let ev = yog_abi::YogBlockBreakEvent {
1696        player: YogStr::from_str(&p), block: YogStr::from_str(&b),
1697        pos: YogBlockPos { x, y, z },
1698    };
1699    let srv = srv_ptr();
1700    guard("on_block_break", || {
1701        for (ud, f) in &handlers().block_break {
1702            unsafe { f(*ud, srv, &ev, 1) };
1703        }
1704    });
1705}
1706
1707#[no_mangle]
1708pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnChat<'l>(
1709    mut env: JNIEnv<'l>, _class: JClass<'l>,
1710    player: JString<'l>, message: JString<'l>,
1711) {
1712    let (p, m) = (jstr!(env, player), jstr!(env, message));
1713    let ev = yog_abi::YogChatEvent { player: YogStr::from_str(&p), message: YogStr::from_str(&m) };
1714    let srv = srv_ptr();
1715    guard("on_chat", || {
1716        for (ud, f) in &handlers().chat {
1717            unsafe { f(*ud, srv, &ev, 1) };
1718        }
1719    });
1720}
1721
1722#[no_mangle]
1723pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerJoin<'l>(
1724    mut env: JNIEnv<'l>, _class: JClass<'l>,
1725    player: JString<'l>, uuid: JString<'l>,
1726) {
1727    let (p, u) = (jstr!(env, player), jstr!(env, uuid));
1728    let ev = yog_abi::YogPlayerEvent { player: YogStr::from_str(&p), uuid: YogStr::from_str(&u) };
1729    let srv = srv_ptr();
1730    guard("on_player_join", || {
1731        for (ud, f) in &handlers().player_join {
1732            unsafe { f(*ud, srv, &ev, 1) };
1733        }
1734    });
1735    // Process startup grants (give items/books on first join).
1736    let h = handlers();
1737    let mut granted = h.startup_granted.lock().expect("startup_granted poisoned");
1738    yog_logging::info!("processing {} startup grants for player {}", h.startup_grants.len(), p);
1739    for sg in &h.startup_grants {
1740        let key = format!("{}::{}", u, sg.id);
1741        if granted.contains_key(&key) {
1742            yog_logging::info!("startup grant {} already granted, skipping", sg.id);
1743            continue;
1744        }
1745        yog_logging::info!("granting startup grant {} with {} items", sg.id, sg.items.len());
1746        for item_id in &sg.items {
1747            let ok = unsafe { srv_give_item(std::ptr::null_mut(), YogStr::from_str(&p), YogStr::from_str(item_id), 1) };
1748            yog_logging::info!("gave {} to {} -> {}", item_id, p, ok);
1749        }
1750        if let Some(book) = &sg.book {
1751            let ok = unsafe { srv_give_item(std::ptr::null_mut(), YogStr::from_str(&p), YogStr::from_str("minecraft:written_book"), 1) };
1752            yog_logging::info!("gave book {} to {} -> {}", book, p, ok);
1753        }
1754        granted.insert(key, true);
1755    }
1756    drop(granted);
1757}
1758
1759#[no_mangle]
1760pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerLeave<'l>(
1761    mut env: JNIEnv<'l>, _class: JClass<'l>,
1762    player: JString<'l>, uuid: JString<'l>,
1763) {
1764    let (p, u) = (jstr!(env, player), jstr!(env, uuid));
1765    let ev = yog_abi::YogPlayerEvent { player: YogStr::from_str(&p), uuid: YogStr::from_str(&u) };
1766    let srv = srv_ptr();
1767    guard("on_player_leave", || {
1768        for (ud, f) in &handlers().player_leave {
1769            unsafe { f(*ud, srv, &ev, 1) };
1770        }
1771    });
1772}
1773
1774#[no_mangle]
1775pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnUseItem<'l>(
1776    mut env: JNIEnv<'l>, _class: JClass<'l>,
1777    player: JString<'l>, item: JString<'l>,
1778) {
1779    let (p, i) = (jstr!(env, player), jstr!(env, item));
1780    let ev = yog_abi::YogUseItemEvent { player: YogStr::from_str(&p), item: YogStr::from_str(&i) };
1781    let srv = srv_ptr();
1782    guard("on_use_item", || {
1783        for (ud, f) in &handlers().use_item {
1784            unsafe { f(*ud, srv, &ev, 1) };
1785        }
1786    });
1787}
1788
1789#[no_mangle]
1790pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnUseBlock<'l>(
1791    mut env: JNIEnv<'l>, _class: JClass<'l>,
1792    player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
1793) {
1794    let (p, b) = (jstr!(env, player), jstr!(env, block));
1795    let ev = yog_abi::YogUseBlockEvent {
1796        player: YogStr::from_str(&p), block: YogStr::from_str(&b),
1797        pos: YogBlockPos { x, y, z },
1798    };
1799    let srv = srv_ptr();
1800    guard("on_use_block", || {
1801        for (ud, f) in &handlers().use_block {
1802            unsafe { f(*ud, srv, &ev, 1) };
1803        }
1804    });
1805}
1806
1807#[no_mangle]
1808pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnAttackEntity<'l>(
1809    mut env: JNIEnv<'l>, _class: JClass<'l>,
1810    player: JString<'l>, target_type: JString<'l>, target_uuid: JString<'l>,
1811) {
1812    let (p, tt, tu) = (jstr!(env, player), jstr!(env, target_type), jstr!(env, target_uuid));
1813    let ev = yog_abi::YogAttackEntityEvent {
1814        player: YogStr::from_str(&p), target_type: YogStr::from_str(&tt), target_uuid: YogStr::from_str(&tu),
1815    };
1816    let srv = srv_ptr();
1817    guard("on_attack_entity", || {
1818        for (ud, f) in &handlers().attack_entity {
1819            unsafe { f(*ud, srv, &ev, 1) };
1820        }
1821    });
1822}
1823
1824#[no_mangle]
1825pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityDamage<'l>(
1826    mut env: JNIEnv<'l>, _class: JClass<'l>,
1827    entity_type: JString<'l>, uuid: JString<'l>, amount: jfloat, source: JString<'l>,
1828) {
1829    let (et, u, s) = (jstr!(env, entity_type), jstr!(env, uuid), jstr!(env, source));
1830    let ev = yog_abi::YogEntityDamageEvent {
1831        entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u),
1832        amount, source: YogStr::from_str(&s),
1833    };
1834    let srv = srv_ptr();
1835    guard("on_entity_damage", || {
1836        for (ud, f) in &handlers().entity_damage {
1837            unsafe { f(*ud, srv, &ev, 1) };
1838        }
1839    });
1840}
1841
1842#[no_mangle]
1843pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityDeath<'l>(
1844    mut env: JNIEnv<'l>, _class: JClass<'l>,
1845    entity_type: JString<'l>, uuid: JString<'l>, source: JString<'l>,
1846) {
1847    let (et, u, s) = (jstr!(env, entity_type), jstr!(env, uuid), jstr!(env, source));
1848    let ev = yog_abi::YogEntityDeathEvent {
1849        entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u), source: YogStr::from_str(&s),
1850    };
1851    let srv = srv_ptr();
1852    guard("on_entity_death", || {
1853        for (ud, f) in &handlers().entity_death {
1854            unsafe { f(*ud, srv, &ev, 1) };
1855        }
1856    });
1857}
1858
1859#[no_mangle]
1860pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnTick<'l>(
1861    _env: JNIEnv<'l>, _class: JClass<'l>,
1862) {
1863    let h = handlers();
1864    let srv = srv_ptr();
1865    guard("on_tick", || {
1866        for (ud, f) in &h.server_tick {
1867            unsafe { f(*ud, srv) };
1868        }
1869    });
1870
1871    // Scheduler — once tasks
1872    {
1873        let mut sched = h.scheduler.lock().expect("scheduler poisoned");
1874        let mut to_fire: Vec<(*mut c_void, YogScheduledFn)> = Vec::new();
1875        let mut remaining = Vec::new();
1876        for task in sched.once_tasks.drain(..) {
1877            if task.delay_remaining == 0 {
1878                to_fire.push((task.ud, task.f));
1879            } else {
1880                remaining.push(OnceTask { delay_remaining: task.delay_remaining - 1, ..task });
1881            }
1882        }
1883        sched.once_tasks = remaining;
1884        drop(sched);
1885        for (ud, f) in to_fire {
1886            guard("schedule_once", || unsafe { f(ud, srv) });
1887        }
1888    }
1889
1890    // Scheduler — repeating tasks
1891    {
1892        let mut sched = h.scheduler.lock().expect("scheduler poisoned");
1893        let mut to_fire: Vec<(*mut c_void, YogScheduledFn)> = Vec::new();
1894        for task in &mut sched.repeating_tasks {
1895            if task.ticks_left == 0 {
1896                to_fire.push((task.ud, task.f));
1897                task.ticks_left = task.period;
1898            } else {
1899                task.ticks_left -= 1;
1900            }
1901        }
1902        drop(sched);
1903        for (ud, f) in to_fire {
1904            guard("schedule_repeating", || unsafe { f(ud, srv) });
1905        }
1906    }
1907}
1908
1909#[no_mangle]
1910pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnServerStarted<'l>(
1911    _env: JNIEnv<'l>, _class: JClass<'l>,
1912) {
1913    let srv = srv_ptr();
1914    guard("on_server_started", || {
1915        for (ud, f) in &handlers().server_started {
1916            unsafe { f(*ud, srv) };
1917        }
1918    });
1919}
1920
1921#[no_mangle]
1922pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnServerStopping<'l>(
1923    _env: JNIEnv<'l>, _class: JClass<'l>,
1924) {
1925    let srv = srv_ptr();
1926    guard("on_server_stopping", || {
1927        for (ud, f) in &handlers().server_stopping {
1928            unsafe { f(*ud, srv) };
1929        }
1930    });
1931}
1932
1933#[no_mangle]
1934pub extern "system" fn Java_dev_yog_NativeBridge_nativeCommandNames<'l>(
1935    env: JNIEnv<'l>, _class: JClass<'l>,
1936) -> jstring {
1937    let names = handlers().commands.keys().cloned().collect::<Vec<_>>().join("\n");
1938    env.new_string(names).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
1939}
1940
1941#[no_mangle]
1942pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnBlockBreakPre<'l>(
1943    mut env: JNIEnv<'l>, _class: JClass<'l>,
1944    player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
1945) -> jni::sys::jboolean {
1946    let h = handlers();
1947    if h.block_break.is_empty() { return 1; }
1948    let p = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
1949    let b = match env.get_string(&block)  { Ok(s) => String::from(s), Err(_) => return 1 };
1950    let ev = yog_abi::YogBlockBreakEvent {
1951        player: YogStr::from_str(&p), block: YogStr::from_str(&b),
1952        pos: YogBlockPos { x, y, z },
1953    };
1954    let srv = srv_ptr();
1955    let mut allow = true;
1956    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
1957        for (ud, f) in &h.block_break {
1958            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
1959        }
1960    })).ok();
1961    allow as jni::sys::jboolean
1962}
1963
1964#[no_mangle]
1965pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnChatPre<'l>(
1966    mut env: JNIEnv<'l>, _class: JClass<'l>,
1967    player: JString<'l>, message: JString<'l>,
1968) -> jni::sys::jboolean {
1969    let h = handlers();
1970    if h.chat.is_empty() { return 1; }
1971    let p = match env.get_string(&player)  { Ok(s) => String::from(s), Err(_) => return 1 };
1972    let m = match env.get_string(&message) { Ok(s) => String::from(s), Err(_) => return 1 };
1973    let ev = yog_abi::YogChatEvent { player: YogStr::from_str(&p), message: YogStr::from_str(&m) };
1974    let srv = srv_ptr();
1975    let mut allow = true;
1976    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
1977        for (ud, f) in &h.chat {
1978            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
1979        }
1980    })).ok();
1981    allow as jni::sys::jboolean
1982}
1983
1984#[no_mangle]
1985pub extern "system" fn Java_dev_yog_NativeBridge_nativeRecipeJsons<'l>(
1986    env: JNIEnv<'l>, _class: JClass<'l>,
1987) -> jstring {
1988    let s = handlers().recipes.iter()
1989        .map(|(ns, name, json)| format!("{}\t{}\t{}", ns, name, json))
1990        .collect::<Vec<_>>().join("\n");
1991    env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
1992}
1993
1994#[no_mangle]
1995pub extern "system" fn Java_dev_yog_NativeBridge_nativeTypedCommandSchemas<'l>(
1996    env: JNIEnv<'l>, _class: JClass<'l>,
1997) -> jstring {
1998    let s = handlers().typed_schemas.iter()
1999        .map(|(name, schema)| format!("{}\t{}", name, schema))
2000        .collect::<Vec<_>>().join("\n");
2001    env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2002}
2003
2004#[no_mangle]
2005pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnCommand<'l>(
2006    mut env: JNIEnv<'l>, _class: JClass<'l>,
2007    name: JString<'l>, args: JString<'l>, source: JString<'l>, uuid: JString<'l>,
2008) -> jstring {
2009    let (n, a, s, u) = (
2010        env.get_string(&name).map(String::from).unwrap_or_default(),
2011        env.get_string(&args).map(String::from).unwrap_or_default(),
2012        env.get_string(&source).map(String::from).unwrap_or_default(),
2013        env.get_string(&uuid).map(String::from).unwrap_or_default(),
2014    );
2015    let ev = yog_abi::YogCommandEvent {
2016        name: YogStr::from_str(&n), args: YogStr::from_str(&a),
2017        source: YogStr::from_str(&s), uuid: YogStr::from_str(&u),
2018    };
2019    let h = handlers();
2020    let srv = srv_ptr();
2021    let reply = std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2022        if let Some((ud, f)) = h.commands.get(&n) {
2023            let mut buf = [0u8; 4096];
2024            let mut reply_len: u32 = 0;
2025            unsafe { f(*ud, srv, &ev, buf.as_mut_ptr(), buf.len() as u32, &mut reply_len) };
2026            String::from_utf8_lossy(&buf[..reply_len as usize]).into_owned()
2027        } else {
2028            String::new()
2029        }
2030    }))
2031    .unwrap_or_else(|_| { yog_logging::error!("a mod panicked handling command `{}`", n); String::new() });
2032
2033    env.new_string(reply).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2034}
2035
2036#[no_mangle]
2037pub extern "system" fn Java_dev_yog_NativeBridge_nativeItemDefs<'l>(
2038    env: JNIEnv<'l>, _class: JClass<'l>,
2039) -> jstring {
2040    let s = handlers().items.iter().map(|d| {
2041        let mut parts = vec![d.id.clone()];
2042        parts.push(format!("max_stack={}", d.max_stack));
2043        if let Some(n) = &d.name    { parts.push(format!("name={n}")); }
2044        if let Some(t) = &d.tooltip { parts.push(format!("tooltip={t}")); }
2045        if d.max_damage > 0         { parts.push(format!("max_damage={}", d.max_damage)); }
2046        if d.fire_resistant         { parts.push("fire_resistant=1".into()); }
2047        if d.fuel_ticks > 0         { parts.push(format!("fuel_ticks={}", d.fuel_ticks)); }
2048        if let Some(f) = &d.food {
2049            parts.push(format!("food={}:{}:{}", f.nutrition, f.saturation, if f.can_always_eat { 1 } else { 0 }));
2050        }
2051        parts.join("\t")
2052    }).collect::<Vec<_>>().join("\n");
2053    env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2054}
2055
2056#[no_mangle]
2057pub extern "system" fn Java_dev_yog_NativeBridge_nativeBlockDefs<'l>(
2058    env: JNIEnv<'l>, _class: JClass<'l>,
2059) -> jstring {
2060    let s = handlers().blocks.iter().map(|d| {
2061        let mut parts = vec![d.id.clone()];
2062        parts.push(format!("hardness={}", d.hardness));
2063        parts.push(format!("resistance={}", d.resistance));
2064        if let Some(n) = &d.name { parts.push(format!("name={n}")); }
2065        if let Some(sh) = d.shape {
2066            parts.push(format!("shape={}:{}:{}:{}:{}:{}", sh[0], sh[1], sh[2], sh[3], sh[4], sh[5]));
2067        }
2068        if d.light_level > 0    { parts.push(format!("light={}", d.light_level)); }
2069        if let Some(snd) = &d.sound { parts.push(format!("sound={snd}")); }
2070        if d.requires_tool       { parts.push("requires_tool=1".into()); }
2071        if d.no_collision        { parts.push("no_collision=1".into()); }
2072        if d.slipperiness > 0.0  { parts.push(format!("slipperiness={}", d.slipperiness)); }
2073        parts.join("\t")
2074    }).collect::<Vec<_>>().join("\n");
2075    env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2076}
2077
2078#[no_mangle]
2079pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPacket<'l>(
2080    mut env: JNIEnv<'l>, _class: JClass<'l>,
2081    channel: JString<'l>, player: JString<'l>, payload: JByteArray<'l>,
2082) {
2083    let ch = env.get_string(&channel).map(String::from).unwrap_or_default();
2084    let pl = env.get_string(&player).map(String::from).unwrap_or_default();
2085    let data = env.convert_byte_array(&payload).unwrap_or_default();
2086    let ev = yog_abi::YogPacketEvent {
2087        channel: YogStr::from_str(&ch), player: YogStr::from_str(&pl),
2088        payload: data.as_ptr(), payload_len: data.len() as u32,
2089    };
2090    let h = handlers();
2091    let srv = srv_ptr();
2092    guard("on_packet", || {
2093        if let Some((ud, f)) = h.packets.get(&ch) {
2094            unsafe { f(*ud, srv, &ev) };
2095        }
2096    });
2097}
2098
2099#[no_mangle]
2100pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnClientPacket<'l>(
2101    mut env: JNIEnv<'l>, _class: JClass<'l>,
2102    channel: JString<'l>, payload: JByteArray<'l>,
2103) {
2104    let ch = env.get_string(&channel).map(String::from).unwrap_or_default();
2105    let data = env.convert_byte_array(&payload).unwrap_or_default();
2106    let ev = yog_abi::YogPacketEvent {
2107        channel: YogStr::from_str(&ch), player: YogStr::EMPTY,
2108        payload: data.as_ptr(), payload_len: data.len() as u32,
2109    };
2110    let h = handlers();
2111    let srv = srv_ptr();
2112    guard("on_client_packet", || {
2113        if let Some((ud, f)) = h.client_packets.get(&ch) {
2114            unsafe { f(*ud, srv, &ev) };
2115        }
2116    });
2117}
2118
2119#[no_mangle]
2120pub extern "system" fn Java_dev_yog_NativeBridge_nativePacketChannels<'l>(
2121    env: JNIEnv<'l>, _class: JClass<'l>,
2122) -> jstring {
2123    let s = handlers().packets.keys().cloned().collect::<Vec<_>>().join("\n");
2124    env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2125}
2126
2127#[no_mangle]
2128pub extern "system" fn Java_dev_yog_NativeBridge_nativeClientPacketChannels<'l>(
2129    env: JNIEnv<'l>, _class: JClass<'l>,
2130) -> jstring {
2131    let s = handlers().client_packets.keys().cloned().collect::<Vec<_>>().join("\n");
2132    env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2133}
2134
2135#[no_mangle]
2136pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntitySpawn<'l>(
2137    mut env: JNIEnv<'l>, _class: JClass<'l>,
2138    entity_type: JString<'l>, uuid: JString<'l>, dimension: JString<'l>,
2139) {
2140    let h = handlers();
2141    if h.entity_spawn.is_empty() { return; }
2142    let (et, u, d) = (jstr!(env, entity_type), jstr!(env, uuid), jstr!(env, dimension));
2143    let ev = yog_abi::YogEntitySpawnEvent {
2144        entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u),
2145        dimension: YogStr::from_str(&d),
2146    };
2147    let srv = srv_ptr();
2148    guard("on_entity_spawn", || {
2149        for (ud, f) in &h.entity_spawn {
2150            unsafe { f(*ud, srv, &ev, 1) };
2151        }
2152    });
2153}
2154
2155#[no_mangle]
2156pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntitySpawnPre<'l>(
2157    mut env: JNIEnv<'l>, _class: JClass<'l>,
2158    entity_type: JString<'l>, uuid: JString<'l>, dimension: JString<'l>,
2159) -> jni::sys::jboolean {
2160    let h = handlers();
2161    if h.entity_spawn.is_empty() { return 1; }
2162    let et = match env.get_string(&entity_type) { Ok(s) => String::from(s), Err(_) => return 1 };
2163    let u  = match env.get_string(&uuid)        { Ok(s) => String::from(s), Err(_) => return 1 };
2164    let d  = match env.get_string(&dimension)   { Ok(s) => String::from(s), Err(_) => return 1 };
2165    let ev = yog_abi::YogEntitySpawnEvent {
2166        entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u),
2167        dimension: YogStr::from_str(&d),
2168    };
2169    let srv = srv_ptr();
2170    let mut allow = true;
2171    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2172        for (ud, f) in &h.entity_spawn {
2173            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2174        }
2175    })).ok();
2176    allow as jni::sys::jboolean
2177}
2178
2179#[no_mangle]
2180pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityDamagePre<'l>(
2181    mut env: JNIEnv<'l>, _class: JClass<'l>,
2182    entity_type: JString<'l>, uuid: JString<'l>, amount: jfloat, source: JString<'l>,
2183) -> jni::sys::jboolean {
2184    let h = handlers();
2185    if h.entity_damage.is_empty() { return 1; }
2186    let et = match env.get_string(&entity_type) { Ok(s) => String::from(s), Err(_) => return 1 };
2187    let u  = match env.get_string(&uuid)        { Ok(s) => String::from(s), Err(_) => return 1 };
2188    let s  = match env.get_string(&source)      { Ok(s) => String::from(s), Err(_) => return 1 };
2189    let ev = yog_abi::YogEntityDamageEvent {
2190        entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u),
2191        amount, source: YogStr::from_str(&s),
2192    };
2193    let srv = srv_ptr();
2194    let mut allow = true;
2195    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2196        for (ud, f) in &h.entity_damage {
2197            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2198        }
2199    })).ok();
2200    allow as jni::sys::jboolean
2201}
2202
2203#[no_mangle]
2204pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlaceBlockPre<'l>(
2205    mut env: JNIEnv<'l>, _class: JClass<'l>,
2206    player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
2207) -> jni::sys::jboolean {
2208    let h = handlers();
2209    if h.player_place_block.is_empty() { return 1; }
2210    let p = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
2211    let b = match env.get_string(&block)  { Ok(s) => String::from(s), Err(_) => return 1 };
2212    let ev = YogPlaceBlockEvent {
2213        player: YogStr::from_str(&p), block: YogStr::from_str(&b),
2214        pos: YogBlockPos { x, y, z },
2215    };
2216    let srv = srv_ptr();
2217    let mut allow = true;
2218    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2219        for (ud, f) in &h.player_place_block {
2220            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2221        }
2222    })).ok();
2223    allow as jni::sys::jboolean
2224}
2225
2226#[no_mangle]
2227pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlaceBlock<'l>(
2228    mut env: JNIEnv<'l>, _class: JClass<'l>,
2229    player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
2230) {
2231    let h = handlers();
2232    if h.player_place_block.is_empty() { return; }
2233    let (p, b) = (jstr!(env, player), jstr!(env, block));
2234    let ev = YogPlaceBlockEvent {
2235        player: YogStr::from_str(&p), block: YogStr::from_str(&b),
2236        pos: YogBlockPos { x, y, z },
2237    };
2238    let srv = srv_ptr();
2239    guard("on_player_place_block", || {
2240        for (ud, f) in &h.player_place_block {
2241            unsafe { f(*ud, srv, &ev, 1) };
2242        }
2243    });
2244}
2245
2246#[no_mangle]
2247pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerDeathPre<'l>(
2248    mut env: JNIEnv<'l>, _class: JClass<'l>,
2249    player: JString<'l>, uuid: JString<'l>, source: JString<'l>,
2250) -> jni::sys::jboolean {
2251    let h = handlers();
2252    if h.player_death.is_empty() { return 1; }
2253    let p  = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
2254    let u  = match env.get_string(&uuid)   { Ok(s) => String::from(s), Err(_) => return 1 };
2255    let s  = match env.get_string(&source) { Ok(s) => String::from(s), Err(_) => return 1 };
2256    let ev = YogPlayerDeathEvent {
2257        player: YogStr::from_str(&p), uuid: YogStr::from_str(&u), source: YogStr::from_str(&s),
2258    };
2259    let srv = srv_ptr();
2260    let mut allow = true;
2261    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2262        for (ud, f) in &h.player_death {
2263            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2264        }
2265    })).ok();
2266    allow as jni::sys::jboolean
2267}
2268
2269#[no_mangle]
2270pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerDeath<'l>(
2271    mut env: JNIEnv<'l>, _class: JClass<'l>,
2272    player: JString<'l>, uuid: JString<'l>, source: JString<'l>,
2273) {
2274    let h = handlers();
2275    if h.player_death.is_empty() { return; }
2276    let (p, u, s) = (jstr!(env, player), jstr!(env, uuid), jstr!(env, source));
2277    let ev = YogPlayerDeathEvent {
2278        player: YogStr::from_str(&p), uuid: YogStr::from_str(&u), source: YogStr::from_str(&s),
2279    };
2280    let srv = srv_ptr();
2281    guard("on_player_death", || {
2282        for (ud, f) in &h.player_death {
2283            unsafe { f(*ud, srv, &ev, 1) };
2284        }
2285    });
2286}
2287
2288#[no_mangle]
2289pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerRespawn<'l>(
2290    mut env: JNIEnv<'l>, _class: JClass<'l>,
2291    player: JString<'l>, uuid: JString<'l>, at_anchor: jni::sys::jboolean,
2292) {
2293    let h = handlers();
2294    if h.player_respawn.is_empty() { return; }
2295    let (p, u) = (jstr!(env, player), jstr!(env, uuid));
2296    let ev = YogPlayerRespawnEvent {
2297        player: YogStr::from_str(&p), uuid: YogStr::from_str(&u), at_anchor: at_anchor != 0,
2298    };
2299    let srv = srv_ptr();
2300    guard("on_player_respawn", || {
2301        for (ud, f) in &h.player_respawn {
2302            unsafe { f(*ud, srv, &ev, 1) };
2303        }
2304    });
2305}
2306
2307#[no_mangle]
2308pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnAdvancement<'l>(
2309    mut env: JNIEnv<'l>, _class: JClass<'l>,
2310    player: JString<'l>, uuid: JString<'l>, advancement: JString<'l>,
2311) {
2312    let h = handlers();
2313    if h.advancement.is_empty() { return; }
2314    let (p, u, a) = (jstr!(env, player), jstr!(env, uuid), jstr!(env, advancement));
2315    let ev = YogAdvancementEvent {
2316        player: YogStr::from_str(&p), uuid: YogStr::from_str(&u), advancement: YogStr::from_str(&a),
2317    };
2318    let srv = srv_ptr();
2319    guard("on_advancement", || {
2320        for (ud, f) in &h.advancement {
2321            unsafe { f(*ud, srv, &ev, 1) };
2322        }
2323    });
2324}
2325
2326#[no_mangle]
2327pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityInteractPre<'l>(
2328    mut env: JNIEnv<'l>, _class: JClass<'l>,
2329    player: JString<'l>, player_uuid: JString<'l>,
2330    entity_type: JString<'l>, entity_uuid: JString<'l>, hand: JString<'l>,
2331) -> jni::sys::jboolean {
2332    let h = handlers();
2333    if h.entity_interact.is_empty() { return 1; }
2334    let p  = match env.get_string(&player)      { Ok(s) => String::from(s), Err(_) => return 1 };
2335    let pu = match env.get_string(&player_uuid)  { Ok(s) => String::from(s), Err(_) => return 1 };
2336    let et = match env.get_string(&entity_type)  { Ok(s) => String::from(s), Err(_) => return 1 };
2337    let eu = match env.get_string(&entity_uuid)  { Ok(s) => String::from(s), Err(_) => return 1 };
2338    let ha = match env.get_string(&hand)         { Ok(s) => String::from(s), Err(_) => return 1 };
2339    let ev = YogEntityInteractEvent {
2340        player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2341        entity_type: YogStr::from_str(&et), entity_uuid: YogStr::from_str(&eu),
2342        hand: YogStr::from_str(&ha),
2343    };
2344    let srv = srv_ptr();
2345    let mut allow = true;
2346    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2347        for (ud, f) in &h.entity_interact {
2348            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2349        }
2350    })).ok();
2351    allow as jni::sys::jboolean
2352}
2353
2354#[no_mangle]
2355pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityInteract<'l>(
2356    mut env: JNIEnv<'l>, _class: JClass<'l>,
2357    player: JString<'l>, player_uuid: JString<'l>,
2358    entity_type: JString<'l>, entity_uuid: JString<'l>, hand: JString<'l>,
2359) {
2360    let h = handlers();
2361    if h.entity_interact.is_empty() { return; }
2362    let (p, pu) = (jstr!(env, player), jstr!(env, player_uuid));
2363    let (et, eu, ha) = (jstr!(env, entity_type), jstr!(env, entity_uuid), jstr!(env, hand));
2364    let ev = YogEntityInteractEvent {
2365        player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2366        entity_type: YogStr::from_str(&et), entity_uuid: YogStr::from_str(&eu),
2367        hand: YogStr::from_str(&ha),
2368    };
2369    let srv = srv_ptr();
2370    guard("on_entity_interact", || {
2371        for (ud, f) in &h.entity_interact {
2372            unsafe { f(*ud, srv, &ev, 1) };
2373        }
2374    });
2375}
2376
2377#[no_mangle]
2378pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnItemCraft<'l>(
2379    mut env: JNIEnv<'l>, _class: JClass<'l>,
2380    player: JString<'l>, player_uuid: JString<'l>,
2381    result_item: JString<'l>, result_count: jint,
2382) {
2383    let h = handlers();
2384    if h.item_craft.is_empty() { return; }
2385    let (p, pu, ri) = (jstr!(env, player), jstr!(env, player_uuid), jstr!(env, result_item));
2386    let ev = YogCraftEvent {
2387        player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2388        result_item: YogStr::from_str(&ri), result_count: result_count as u32,
2389    };
2390    let srv = srv_ptr();
2391    guard("on_item_craft", || {
2392        for (ud, f) in &h.item_craft {
2393            unsafe { f(*ud, srv, &ev, 1) };
2394        }
2395    });
2396}
2397
2398#[no_mangle]
2399pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnExplosionPre<'l>(
2400    mut env: JNIEnv<'l>, _class: JClass<'l>,
2401    dimension: JString<'l>, x: jdouble, y: jdouble, z: jdouble,
2402    power: jfloat, cause_uuid: JString<'l>,
2403) -> jni::sys::jboolean {
2404    let h = handlers();
2405    if h.explosion.is_empty() { return 1; }
2406    let d  = match env.get_string(&dimension)  { Ok(s) => String::from(s), Err(_) => return 1 };
2407    let cu = match env.get_string(&cause_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2408    let ev = YogExplosionEvent {
2409        dimension: YogStr::from_str(&d), x, y, z, power, cause_uuid: YogStr::from_str(&cu),
2410    };
2411    let srv = srv_ptr();
2412    let mut allow = true;
2413    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2414        for (ud, f) in &h.explosion {
2415            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2416        }
2417    })).ok();
2418    allow as jni::sys::jboolean
2419}
2420
2421#[no_mangle]
2422pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnExplosion<'l>(
2423    mut env: JNIEnv<'l>, _class: JClass<'l>,
2424    dimension: JString<'l>, x: jdouble, y: jdouble, z: jdouble,
2425    power: jfloat, cause_uuid: JString<'l>,
2426) {
2427    let h = handlers();
2428    if h.explosion.is_empty() { return; }
2429    let (d, cu) = (jstr!(env, dimension), jstr!(env, cause_uuid));
2430    let ev = YogExplosionEvent {
2431        dimension: YogStr::from_str(&d), x, y, z, power, cause_uuid: YogStr::from_str(&cu),
2432    };
2433    let srv = srv_ptr();
2434    guard("on_explosion", || {
2435        for (ud, f) in &h.explosion {
2436            unsafe { f(*ud, srv, &ev, 1) };
2437        }
2438    });
2439}
2440
2441// ── ABI minor 9 JNI entry points ─────────────────────────────────────────────
2442
2443#[no_mangle]
2444pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnItemPickupPre<'l>(
2445    mut env: JNIEnv<'l>, _class: JClass<'l>,
2446    player: JString<'l>, player_uuid: JString<'l>,
2447    item_id: JString<'l>, item_count: jint, entity_uuid: JString<'l>,
2448) -> jni::sys::jboolean {
2449    let h = handlers();
2450    if h.item_pickup.is_empty() { return 1; }
2451    let p   = match env.get_string(&player)      { Ok(s) => String::from(s), Err(_) => return 1 };
2452    let pu  = match env.get_string(&player_uuid)  { Ok(s) => String::from(s), Err(_) => return 1 };
2453    let ii  = match env.get_string(&item_id)      { Ok(s) => String::from(s), Err(_) => return 1 };
2454    let eu  = match env.get_string(&entity_uuid)  { Ok(s) => String::from(s), Err(_) => return 1 };
2455    let ev = YogItemPickupEvent {
2456        player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2457        item_id: YogStr::from_str(&ii), item_count: item_count as u32,
2458        entity_uuid: YogStr::from_str(&eu),
2459    };
2460    let srv = srv_ptr();
2461    let mut allow = true;
2462    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2463        for (ud, f) in &h.item_pickup {
2464            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2465        }
2466    })).ok();
2467    allow as jni::sys::jboolean
2468}
2469
2470#[no_mangle]
2471pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnItemPickup<'l>(
2472    mut env: JNIEnv<'l>, _class: JClass<'l>,
2473    player: JString<'l>, player_uuid: JString<'l>,
2474    item_id: JString<'l>, item_count: jint, entity_uuid: JString<'l>,
2475) {
2476    let h = handlers();
2477    if h.item_pickup.is_empty() { return; }
2478    let (p, pu) = (jstr!(env, player), jstr!(env, player_uuid));
2479    let (ii, eu) = (jstr!(env, item_id), jstr!(env, entity_uuid));
2480    let ev = YogItemPickupEvent {
2481        player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2482        item_id: YogStr::from_str(&ii), item_count: item_count as u32,
2483        entity_uuid: YogStr::from_str(&eu),
2484    };
2485    let srv = srv_ptr();
2486    guard("on_item_pickup", || {
2487        for (ud, f) in &h.item_pickup {
2488            unsafe { f(*ud, srv, &ev, 1) };
2489        }
2490    });
2491}
2492
2493#[no_mangle]
2494pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerMove<'l>(
2495    mut env: JNIEnv<'l>, _class: JClass<'l>,
2496    player: JString<'l>, player_uuid: JString<'l>,
2497    x: jdouble, y: jdouble, z: jdouble, yaw: jfloat, pitch: jfloat,
2498) {
2499    let h = handlers();
2500    if h.player_move.is_empty() { return; }
2501    let (p, pu) = (jstr!(env, player), jstr!(env, player_uuid));
2502    let ev = YogPlayerMoveEvent {
2503        player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2504        x, y, z, yaw, pitch,
2505    };
2506    let srv = srv_ptr();
2507    guard("on_player_move", || {
2508        for (ud, f) in &h.player_move {
2509            unsafe { f(*ud, srv, &ev, 1) };
2510        }
2511    });
2512}
2513
2514#[no_mangle]
2515pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnContainerOpenPre<'l>(
2516    mut env: JNIEnv<'l>, _class: JClass<'l>,
2517    player: JString<'l>, player_uuid: JString<'l>,
2518) -> jni::sys::jboolean {
2519    let h = handlers();
2520    if h.container_open.is_empty() { return 1; }
2521    let p  = match env.get_string(&player)      { Ok(s) => String::from(s), Err(_) => return 1 };
2522    let pu = match env.get_string(&player_uuid)  { Ok(s) => String::from(s), Err(_) => return 1 };
2523    let ev = YogContainerOpenEvent {
2524        player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2525        container_type: YogStr::EMPTY,
2526    };
2527    let srv = srv_ptr();
2528    let mut allow = true;
2529    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2530        for (ud, f) in &h.container_open {
2531            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2532        }
2533    })).ok();
2534    allow as jni::sys::jboolean
2535}
2536
2537#[no_mangle]
2538pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnContainerOpen<'l>(
2539    mut env: JNIEnv<'l>, _class: JClass<'l>,
2540    player: JString<'l>, player_uuid: JString<'l>, container_type: JString<'l>,
2541) {
2542    let h = handlers();
2543    if h.container_open.is_empty() { return; }
2544    let (p, pu, ct) = (jstr!(env, player), jstr!(env, player_uuid), jstr!(env, container_type));
2545    let ev = YogContainerOpenEvent {
2546        player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2547        container_type: YogStr::from_str(&ct),
2548    };
2549    let srv = srv_ptr();
2550    guard("on_container_open", || {
2551        for (ud, f) in &h.container_open {
2552            unsafe { f(*ud, srv, &ev, 1) };
2553        }
2554    });
2555}
2556
2557#[no_mangle]
2558pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnContainerClose<'l>(
2559    mut env: JNIEnv<'l>, _class: JClass<'l>,
2560    player: JString<'l>, player_uuid: JString<'l>,
2561) {
2562    let h = handlers();
2563    if h.container_close.is_empty() { return; }
2564    let (p, pu) = (jstr!(env, player), jstr!(env, player_uuid));
2565    let ev = YogContainerCloseEvent {
2566        player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2567    };
2568    let srv = srv_ptr();
2569    guard("on_container_close", || {
2570        for (ud, f) in &h.container_close {
2571            unsafe { f(*ud, srv, &ev, 1) };
2572        }
2573    });
2574}
2575
2576#[no_mangle]
2577pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnProjectileHitPre<'l>(
2578    mut env: JNIEnv<'l>, _class: JClass<'l>,
2579    projectile_type: JString<'l>, projectile_uuid: JString<'l>, shooter_uuid: JString<'l>,
2580    hit_type: JString<'l>, hit_entity_uuid: JString<'l>,
2581    x: jdouble, y: jdouble, z: jdouble, dimension: JString<'l>,
2582) -> jni::sys::jboolean {
2583    let h = handlers();
2584    if h.projectile_hit.is_empty() { return 1; }
2585    let pt  = match env.get_string(&projectile_type)  { Ok(s) => String::from(s), Err(_) => return 1 };
2586    let pu  = match env.get_string(&projectile_uuid)  { Ok(s) => String::from(s), Err(_) => return 1 };
2587    let su  = match env.get_string(&shooter_uuid)     { Ok(s) => String::from(s), Err(_) => return 1 };
2588    let ht  = match env.get_string(&hit_type)         { Ok(s) => String::from(s), Err(_) => return 1 };
2589    let heu = match env.get_string(&hit_entity_uuid)  { Ok(s) => String::from(s), Err(_) => return 1 };
2590    let dim = match env.get_string(&dimension)        { Ok(s) => String::from(s), Err(_) => return 1 };
2591    let ev = YogProjectileHitEvent {
2592        projectile_type: YogStr::from_str(&pt), projectile_uuid: YogStr::from_str(&pu),
2593        shooter_uuid: YogStr::from_str(&su), hit_type: YogStr::from_str(&ht),
2594        hit_entity_uuid: YogStr::from_str(&heu), x, y, z, dimension: YogStr::from_str(&dim),
2595    };
2596    let srv = srv_ptr();
2597    let mut allow = true;
2598    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2599        for (ud, f) in &h.projectile_hit {
2600            if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2601        }
2602    })).ok();
2603    allow as jni::sys::jboolean
2604}
2605
2606#[no_mangle]
2607pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnProjectileHit<'l>(
2608    mut env: JNIEnv<'l>, _class: JClass<'l>,
2609    projectile_type: JString<'l>, projectile_uuid: JString<'l>, shooter_uuid: JString<'l>,
2610    hit_type: JString<'l>, hit_entity_uuid: JString<'l>,
2611    x: jdouble, y: jdouble, z: jdouble, dimension: JString<'l>,
2612) {
2613    let h = handlers();
2614    if h.projectile_hit.is_empty() { return; }
2615    let (pt, pu) = (jstr!(env, projectile_type), jstr!(env, projectile_uuid));
2616    let (su, ht) = (jstr!(env, shooter_uuid), jstr!(env, hit_type));
2617    let (heu, dim) = (jstr!(env, hit_entity_uuid), jstr!(env, dimension));
2618    let ev = YogProjectileHitEvent {
2619        projectile_type: YogStr::from_str(&pt), projectile_uuid: YogStr::from_str(&pu),
2620        shooter_uuid: YogStr::from_str(&su), hit_type: YogStr::from_str(&ht),
2621        hit_entity_uuid: YogStr::from_str(&heu), x, y, z, dimension: YogStr::from_str(&dim),
2622    };
2623    let srv = srv_ptr();
2624    guard("on_projectile_hit", || {
2625        for (ud, f) in &h.projectile_hit {
2626            unsafe { f(*ud, srv, &ev, 1) };
2627        }
2628    });
2629}
2630
2631// ── ABI minor 10 — client-side JNI entry points ───────────────────────────────
2632
2633#[no_mangle]
2634pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnClientTick<'l>(
2635    _env: JNIEnv<'l>, _class: JClass<'l>,
2636) {
2637    let h = handlers();
2638    if h.client_tick.is_empty() { return; }
2639    guard("on_client_tick", || {
2640        for (ud, f) in &h.client_tick {
2641            unsafe { f(*ud) };
2642        }
2643    });
2644}
2645
2646#[no_mangle]
2647pub extern "system" fn Java_dev_yog_NativeBridge_nativeGlInit<'l>(
2648    _env: JNIEnv<'l>, _class: JClass<'l>,
2649) {
2650    if GL.get().is_some() { return; }
2651    let mut raw_get_binary: usize = 0;
2652    let mut raw_prog_binary: usize = 0;
2653    let gl = unsafe {
2654        glow::Context::from_loader_function(|sym| {
2655            let Some(mut env) = get_env() else { return std::ptr::null() };
2656            let jsym = match env.new_string(sym) {
2657                Ok(s) => s,
2658                Err(_) => return std::ptr::null(),
2659            };
2660            let jsym_obj: JObject = jsym.into();
2661            let val = env.call_static_method(
2662                "dev/yog/NativeBridge",
2663                "glProcAddress",
2664                "(Ljava/lang/String;)J",
2665                &[JValue::Object(&jsym_obj)],
2666            );
2667            let ptr = match val.and_then(|v| v.j()) {
2668                Ok(p) if p != 0 => p as usize as *const _,
2669                _ => std::ptr::null(),
2670            };
2671            // Capture extension pointers while the loader runs.
2672            match sym {
2673                "glGetProgramBinary" => raw_get_binary = ptr as usize,
2674                "glProgramBinary"    => raw_prog_binary = ptr as usize,
2675                _ => {}
2676            }
2677            ptr
2678        })
2679    };
2680    let _ = GL.set(GlCtx(gl));
2681    let _ = GL_GET_PROGRAM_BINARY.set(if raw_get_binary != 0 { Some(raw_get_binary) } else { None });
2682    let _ = GL_PROGRAM_BINARY.set(if raw_prog_binary != 0 { Some(raw_prog_binary) } else { None });
2683    // `glGetProgramiv` is a core function; always available.  We look it up once here
2684    // to avoid depending on glow internals for the PROGRAM_BINARY_LENGTH query.
2685    if let Some(mut env) = get_env() {
2686        if let Ok(jsym) = env.new_string("glGetProgramiv") {
2687            let jsym_obj: JObject = jsym.into();
2688            if let Ok(jv) = env.call_static_method(
2689                "dev/yog/NativeBridge", "glProcAddress", "(Ljava/lang/String;)J",
2690                &[JValue::Object(&jsym_obj)],
2691            ) {
2692                if let Ok(ptr) = jv.j() {
2693                    let _ = GL_GET_PROGRAM_IV.set(if ptr != 0 { Some(ptr as usize) } else { None });
2694                }
2695            }
2696        }
2697    }
2698}
2699
2700#[no_mangle]
2701pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnHudRender<'l>(
2702    _env: JNIEnv<'l>, _class: JClass<'l>,
2703    delta_tick: jfloat,
2704    screen_w: jint,
2705    screen_h: jint,
2706    scale_factor: jfloat,
2707    player_x: jfloat, player_y: jfloat, player_z: jfloat,
2708) {
2709    let h = handlers();
2710    if h.hud_render.is_empty() { return; }
2711    let mut gfx = GFX_FN_TABLE;
2712    gfx.screen_w = screen_w;
2713    gfx.screen_h = screen_h;
2714    gfx.delta_tick = delta_tick;
2715    gfx.scale_factor = scale_factor;
2716    gfx.player_pos = [player_x, player_y, player_z];
2717    guard("on_hud_render", || {
2718        for (ud, f) in &h.hud_render {
2719            unsafe { f(*ud, &gfx) };
2720        }
2721    });
2722}
2723
2724#[no_mangle]
2725pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnWorldRender<'l>(
2726    env: JNIEnv<'l>, _class: JClass<'l>,
2727    delta_tick: jfloat,
2728    screen_w: jint,
2729    screen_h: jint,
2730    scale_factor: jfloat,
2731    view_proj_arr: JFloatArray<'l>,
2732    cam_x: jfloat, cam_y: jfloat, cam_z: jfloat,
2733    player_x: jfloat, player_y: jfloat, player_z: jfloat,
2734) {
2735    let h = handlers();
2736    if h.world_render.is_empty() { return; }
2737    let mut view_proj = [0f32; 16];
2738    if env.get_float_array_region(&view_proj_arr, 0, &mut view_proj).is_err() { return; }
2739    let mut gfx = GFX_FN_TABLE;
2740    gfx.screen_w = screen_w;
2741    gfx.screen_h = screen_h;
2742    gfx.delta_tick = delta_tick;
2743    gfx.scale_factor = scale_factor;
2744    gfx.view_proj = view_proj;
2745    gfx.camera_pos = [cam_x, cam_y, cam_z];
2746    gfx.player_pos = [player_x, player_y, player_z];
2747    guard("on_world_render", || {
2748        for (ud, f) in &h.world_render {
2749            unsafe { f(*ud, &gfx) };
2750        }
2751    });
2752}
2753
2754#[no_mangle]
2755pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnKeyPress<'l>(
2756    _env: JNIEnv<'l>, _class: JClass<'l>,
2757    key_code: jint, scan_code: jint, action: jint, modifiers: jint,
2758) -> jni::sys::jboolean {
2759    let h = handlers();
2760    if h.key_press.is_empty() { return 1; }
2761    let ev = YogKeyPressEvent { key_code, scan_code, action, modifiers };
2762    let mut allow = true;
2763    std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2764        for (ud, f) in &h.key_press {
2765            if !unsafe { f(*ud, &ev) } { allow = false; break; }
2766        }
2767    })).ok();
2768    allow as jni::sys::jboolean
2769}
2770
2771#[no_mangle]
2772pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnScreenOpen<'l>(
2773    mut env: JNIEnv<'l>, _class: JClass<'l>,
2774    screen_class: JString<'l>,
2775) {
2776    let h = handlers();
2777    if h.screen_open.is_empty() { return; }
2778    let sc = match env.get_string(&screen_class) { Ok(s) => String::from(s), Err(_) => return };
2779    guard("on_screen_open", || {
2780        for (ud, f) in &h.screen_open {
2781            unsafe { f(*ud, YogStr::from_str(&sc)) };
2782        }
2783    });
2784}
2785
2786#[no_mangle]
2787pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnScreenClose<'l>(
2788    mut env: JNIEnv<'l>, _class: JClass<'l>,
2789    screen_class: JString<'l>,
2790) {
2791    let h = handlers();
2792    if h.screen_close.is_empty() { return; }
2793    let sc = match env.get_string(&screen_class) { Ok(s) => String::from(s), Err(_) => return };
2794    guard("on_screen_close", || {
2795        for (ud, f) in &h.screen_close {
2796            unsafe { f(*ud, YogStr::from_str(&sc)) };
2797        }
2798    });
2799}