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