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