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