Skip to main content

yog_runtime/
lib.rs

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