1use 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,
38 YogUseBlockFn, YogUseItemFn, YogVec3, YogWorldRenderFn,
39};
40use yog_registry::{BlockDef, FoodDef, ItemDef};
41
42static JAVA_VM: OnceLock<JavaVM> = OnceLock::new();
46static LOADED_MODS: Mutex<Vec<Library>> = Mutex::new(Vec::new());
48static SERVER: OnceLock<YogServer> = OnceLock::new();
50static HANDLERS: OnceLock<RuntimeHandlers> = OnceLock::new();
52
53struct GlCtx(glow::Context);
56unsafe impl Send for GlCtx {}
57unsafe impl Sync for GlCtx {}
58
59static GL: OnceLock<GlCtx> = OnceLock::new();
61
62static 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
69struct 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
115unsafe 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
168fn 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
182unsafe 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
193unsafe 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
209unsafe 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
816fn 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
855unsafe 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
889unsafe 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
930fn 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
947unsafe 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
964unsafe 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 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 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 let _ = std::fs::remove_file(path);
1004 }
1005 }
1006
1007 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 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
1091unsafe 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
1140unsafe 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 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
1165unsafe 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
1202unsafe 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
1243static GFX_FN_TABLE: YogGfxApi = YogGfxApi {
1246 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
1282macro_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
1404fn build_server_table() -> YogServer {
1407 YogServer {
1408 ctx: std::ptr::null_mut(),
1409 abi_version: ABI_VERSION,
1410 size: std::mem::size_of::<YogServer>() as u32,
1411 free_str: yog_free_str,
1412 broadcast: srv_broadcast,
1413 get_block: srv_get_block,
1414 set_block: srv_set_block,
1415 world_time: srv_world_time,
1416 set_time: srv_set_time,
1417 is_raining: srv_is_raining,
1418 set_weather: srv_set_weather,
1419 give_item: srv_give_item,
1420 player_teleport: srv_player_teleport,
1421 send_to_player: srv_send_to_player,
1422 send_to_server: srv_send_to_server,
1423 kick_player: srv_kick_player,
1424 set_gamemode: srv_set_gamemode,
1425 send_title: srv_send_title,
1426 send_actionbar: srv_send_actionbar,
1427 play_sound: srv_play_sound,
1428 play_sound_player: srv_play_sound_player,
1429 entity_teleport: srv_entity_teleport,
1430 entity_position: srv_entity_position,
1431 entity_health: srv_entity_health,
1432 entity_set_health: srv_entity_set_health,
1433 entity_kill: srv_entity_kill,
1434 spawn_entity: srv_spawn_entity,
1435 entity_add_effect: srv_entity_add_effect,
1436 entity_remove_effect: srv_entity_remove_effect,
1437 entity_clear_effects: srv_entity_clear_effects,
1438 entity_velocity: srv_entity_velocity,
1439 entity_set_velocity: srv_entity_set_velocity,
1440 entity_add_velocity: srv_entity_add_velocity,
1441 has_item_tag: srv_has_item_tag,
1442 has_block_tag: srv_has_block_tag,
1443 drop_loot: srv_drop_loot,
1444 scoreboard_get: srv_scoreboard_get,
1445 scoreboard_set: srv_scoreboard_set,
1446 scoreboard_add: srv_scoreboard_add,
1447 bossbar_create: srv_bossbar_create,
1448 bossbar_remove: srv_bossbar_remove,
1449 bossbar_set_title: srv_bossbar_set_title,
1450 bossbar_set_progress: srv_bossbar_set_progress,
1451 bossbar_set_color: srv_bossbar_set_color,
1452 bossbar_add_player: srv_bossbar_add_player,
1453 bossbar_remove_player: srv_bossbar_remove_player,
1454 bossbar_set_visible: srv_bossbar_set_visible,
1455 game_dir: srv_game_dir,
1456 get_block_nbt: srv_get_block_nbt,
1457 set_block_nbt: srv_set_block_nbt,
1458 player_inventory: srv_player_inventory,
1459 player_set_slot: srv_player_set_slot,
1460 player_teleport_dim: srv_player_teleport_dim,
1461 entity_teleport_dim: srv_entity_teleport_dim,
1462 online_players: srv_online_players,
1463 world_entity_count: srv_world_entity_count,
1464 entity_get_nbt: srv_entity_get_nbt,
1465 entity_set_nbt: srv_entity_set_nbt,
1466 spawn_particles: srv_spawn_particles,
1467 entity_attribute_get: srv_entity_attribute_get,
1468 entity_attribute_set: srv_entity_attribute_set,
1469 get_held_item_nbt: srv_get_held_item_nbt,
1470 set_held_item_nbt: srv_set_held_item_nbt,
1471 get_offhand_item_nbt: srv_get_offhand_item_nbt,
1472 set_offhand_item_nbt: srv_set_offhand_item_nbt,
1473 get_slot_item: srv_get_slot_item,
1474 set_slot_item: srv_set_slot_item,
1475 }
1476}
1477
1478fn build_api_table(ctx: *mut RuntimeHandlers, server: *const YogServer) -> YogApi {
1479 YogApi {
1480 abi_version: ABI_VERSION,
1481 size: std::mem::size_of::<YogApi>() as u32,
1482 ctx: ctx as *mut c_void,
1483 server,
1484 on_block_break: api_on_block_break,
1485 on_chat: api_on_chat,
1486 on_player_join: api_on_player_join,
1487 on_player_leave: api_on_player_leave,
1488 on_use_item: api_on_use_item,
1489 on_use_block: api_on_use_block,
1490 on_attack_entity: api_on_attack_entity,
1491 on_entity_damage: api_on_entity_damage,
1492 on_entity_death: api_on_entity_death,
1493 on_entity_spawn: api_on_entity_spawn,
1494 on_player_place_block: api_on_player_place_block,
1495 on_player_death: api_on_player_death,
1496 on_player_respawn: api_on_player_respawn,
1497 on_advancement: api_on_advancement,
1498 on_entity_interact: api_on_entity_interact,
1499 on_item_craft: api_on_item_craft,
1500 on_explosion: api_on_explosion,
1501 on_item_pickup: api_on_item_pickup,
1502 on_player_move: api_on_player_move,
1503 on_container_open: api_on_container_open,
1504 on_container_close: api_on_container_close,
1505 on_projectile_hit: api_on_projectile_hit,
1506 on_server_tick: api_on_server_tick,
1507 on_server_started: api_on_server_started,
1508 on_server_stopping: api_on_server_stopping,
1509 on_packet: api_on_packet,
1510 on_client_packet: api_on_client_packet,
1511 register_command: api_register_command,
1512 register_typed_command: api_register_typed_command,
1513 register_recipe_json: api_register_recipe_json,
1514 register_item: api_register_item,
1515 register_block: api_register_block,
1516 schedule_once: api_schedule_once,
1517 schedule_repeating: api_schedule_repeating,
1518 on_client_tick: api_on_client_tick,
1519 on_hud_render: api_on_hud_render,
1520 on_key_press: api_on_key_press,
1521 on_screen_open: api_on_screen_open,
1522 on_screen_close: api_on_screen_close,
1523 on_world_render: api_on_world_render,
1524 }
1525}
1526
1527fn platform_tag() -> String {
1530 format!("{}-{}", std::env::consts::OS, std::env::consts::ARCH)
1531}
1532
1533type AbiVersionFn = unsafe extern "C" fn() -> u32;
1534type RegisterFn = unsafe extern "C" fn(*const YogApi, *mut c_void);
1535
1536fn load_mods(dir: &Path, api: &YogApi) {
1537 let entries = match std::fs::read_dir(dir) {
1538 Ok(e) => e,
1539 Err(_) => {
1540 yog_logging::info!("no mods directory at {} — none loaded", dir.display());
1541 return;
1542 }
1543 };
1544 let mut count = 0u32;
1545 for entry in entries.flatten() {
1546 let path = entry.path();
1547 let lib_path = match path.extension().and_then(|e| e.to_str()) {
1548 Some("yog") => match extract_yog(&path) {
1549 Some(p) => p,
1550 None => {
1551 yog_logging::error!("no native for {} in {}", platform_tag(), path.display());
1552 continue;
1553 }
1554 },
1555 Some("so") | Some("dll") | Some("dylib") => path.clone(),
1556 _ => continue,
1557 };
1558 if load_mod_lib(&lib_path, api) { count += 1; }
1559 }
1560 yog_logging::info!("loaded {} mod(s) from {}", count, dir.display());
1561}
1562
1563fn load_mod_lib(path: &Path, api: &YogApi) -> bool {
1564 unsafe {
1565 let lib = match Library::new(path) {
1566 Ok(l) => l,
1567 Err(e) => { yog_logging::error!("failed to load {}: {}", path.display(), e); return false; }
1568 };
1569 let abi: Symbol<AbiVersionFn> = match lib.get(b"yog_abi_version") {
1570 Ok(s) => s,
1571 Err(_) => { yog_logging::error!("{} is not a Yog mod (no yog_abi_version)", path.display()); return false; }
1572 };
1573 let mod_abi = abi();
1574 if mod_abi != ABI_VERSION {
1575 yog_logging::error!("{}: ABI {} incompatible with runtime ABI {}", path.display(), mod_abi, ABI_VERSION);
1576 return false;
1577 }
1578 let register: Symbol<RegisterFn> = match lib.get(b"yog_mod_register") {
1579 Ok(s) => s,
1580 Err(_) => { yog_logging::error!("{} missing yog_mod_register", path.display()); return false; }
1581 };
1582 register(api as *const YogApi, std::ptr::null_mut());
1583 drop(register);
1584 drop(abi);
1585 LOADED_MODS.lock().expect("mods lock poisoned").push(lib);
1586 }
1587 true
1588}
1589
1590fn extract_yog(path: &Path) -> Option<PathBuf> {
1591 let file = std::fs::File::open(path).ok()?;
1592 let mut archive = zip::ZipArchive::new(file).ok()?;
1593 let prefix = format!("natives/{}/", platform_tag());
1594 let mut entry_name = None;
1595 for i in 0..archive.len() {
1596 let f = archive.by_index(i).ok()?;
1597 if f.name().starts_with(&prefix) && !f.name().ends_with('/') {
1598 entry_name = Some(f.name().to_string());
1599 break;
1600 }
1601 }
1602 let entry_name = entry_name?;
1603 let ext = Path::new(&entry_name).extension().and_then(|e| e.to_str()).unwrap_or("bin");
1604 let stem = path.file_stem()?.to_string_lossy().into_owned();
1605 let out = std::env::temp_dir().join(format!("yog-{}-{}.{}", stem, std::process::id(), ext));
1606 let mut entry = archive.by_name(&entry_name).ok()?;
1607 let mut out_file = std::fs::File::create(&out).ok()?;
1608 std::io::copy(&mut entry, &mut out_file).ok()?;
1609 Some(out)
1610}
1611
1612fn srv_ptr() -> *const YogServer {
1615 SERVER.get().expect("yog: SERVER not initialised") as *const YogServer
1616}
1617
1618#[no_mangle]
1621pub extern "system" fn Java_dev_yog_NativeBridge_nativeInit<'l>(
1622 mut env: JNIEnv<'l>,
1623 _class: JClass<'l>,
1624 mods_dir: JString<'l>,
1625) {
1626 if let Ok(vm) = env.get_java_vm() { let _ = JAVA_VM.set(vm); }
1627
1628 let dir = env.get_string(&mods_dir).map(String::from).unwrap_or_default();
1629
1630 let _ = SERVER.set(build_server_table());
1632 let server_ptr = SERVER.get().unwrap() as *const YogServer;
1633
1634 let mut handlers = Box::new(RuntimeHandlers::new());
1637 let handlers_ptr = &mut *handlers as *mut RuntimeHandlers;
1638
1639 let api = build_api_table(handlers_ptr, server_ptr);
1640
1641 guard("mod loading", || {
1642 load_mods(Path::new(&dir), &api);
1643 });
1644
1645 let _ = HANDLERS.set(*handlers);
1647
1648 yog_logging::info!("runtime initialised — the gate is open.");
1649}
1650
1651#[no_mangle]
1652pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnBlockBreak<'l>(
1653 mut env: JNIEnv<'l>, _class: JClass<'l>,
1654 player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
1655) {
1656 let (p, b) = (jstr!(env, player), jstr!(env, block));
1657 let ev = yog_abi::YogBlockBreakEvent {
1658 player: YogStr::from_str(&p), block: YogStr::from_str(&b),
1659 pos: YogBlockPos { x, y, z },
1660 };
1661 let srv = srv_ptr();
1662 guard("on_block_break", || {
1663 for (ud, f) in &handlers().block_break {
1664 unsafe { f(*ud, srv, &ev, 1) };
1665 }
1666 });
1667}
1668
1669#[no_mangle]
1670pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnChat<'l>(
1671 mut env: JNIEnv<'l>, _class: JClass<'l>,
1672 player: JString<'l>, message: JString<'l>,
1673) {
1674 let (p, m) = (jstr!(env, player), jstr!(env, message));
1675 let ev = yog_abi::YogChatEvent { player: YogStr::from_str(&p), message: YogStr::from_str(&m) };
1676 let srv = srv_ptr();
1677 guard("on_chat", || {
1678 for (ud, f) in &handlers().chat {
1679 unsafe { f(*ud, srv, &ev, 1) };
1680 }
1681 });
1682}
1683
1684#[no_mangle]
1685pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerJoin<'l>(
1686 mut env: JNIEnv<'l>, _class: JClass<'l>,
1687 player: JString<'l>, uuid: JString<'l>,
1688) {
1689 let (p, u) = (jstr!(env, player), jstr!(env, uuid));
1690 let ev = yog_abi::YogPlayerEvent { player: YogStr::from_str(&p), uuid: YogStr::from_str(&u) };
1691 let srv = srv_ptr();
1692 guard("on_player_join", || {
1693 for (ud, f) in &handlers().player_join {
1694 unsafe { f(*ud, srv, &ev, 1) };
1695 }
1696 });
1697 let h = handlers();
1699 let mut granted = h.startup_granted.lock().expect("startup_granted poisoned");
1700 for sg in &h.startup_grants {
1701 let key = format!("{}::{}", u, sg.id);
1702 if granted.contains_key(&key) {
1703 continue;
1704 }
1705 for item_id in &sg.items {
1706 unsafe { srv_give_item(std::ptr::null_mut(), YogStr::from_str(&p), YogStr::from_str(item_id), 1); }
1707 }
1708 if let Some(book) = &sg.book {
1709 unsafe { srv_give_item(std::ptr::null_mut(), YogStr::from_str(&p), YogStr::from_str("minecraft:written_book"), 1); }
1710 }
1711 granted.insert(key, true);
1712 }
1713 drop(granted);
1714}
1715
1716#[no_mangle]
1717pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerLeave<'l>(
1718 mut env: JNIEnv<'l>, _class: JClass<'l>,
1719 player: JString<'l>, uuid: JString<'l>,
1720) {
1721 let (p, u) = (jstr!(env, player), jstr!(env, uuid));
1722 let ev = yog_abi::YogPlayerEvent { player: YogStr::from_str(&p), uuid: YogStr::from_str(&u) };
1723 let srv = srv_ptr();
1724 guard("on_player_leave", || {
1725 for (ud, f) in &handlers().player_leave {
1726 unsafe { f(*ud, srv, &ev, 1) };
1727 }
1728 });
1729}
1730
1731#[no_mangle]
1732pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnUseItem<'l>(
1733 mut env: JNIEnv<'l>, _class: JClass<'l>,
1734 player: JString<'l>, item: JString<'l>,
1735) {
1736 let (p, i) = (jstr!(env, player), jstr!(env, item));
1737 let ev = yog_abi::YogUseItemEvent { player: YogStr::from_str(&p), item: YogStr::from_str(&i) };
1738 let srv = srv_ptr();
1739 guard("on_use_item", || {
1740 for (ud, f) in &handlers().use_item {
1741 unsafe { f(*ud, srv, &ev, 1) };
1742 }
1743 });
1744}
1745
1746#[no_mangle]
1747pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnUseBlock<'l>(
1748 mut env: JNIEnv<'l>, _class: JClass<'l>,
1749 player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
1750) {
1751 let (p, b) = (jstr!(env, player), jstr!(env, block));
1752 let ev = yog_abi::YogUseBlockEvent {
1753 player: YogStr::from_str(&p), block: YogStr::from_str(&b),
1754 pos: YogBlockPos { x, y, z },
1755 };
1756 let srv = srv_ptr();
1757 guard("on_use_block", || {
1758 for (ud, f) in &handlers().use_block {
1759 unsafe { f(*ud, srv, &ev, 1) };
1760 }
1761 });
1762}
1763
1764#[no_mangle]
1765pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnAttackEntity<'l>(
1766 mut env: JNIEnv<'l>, _class: JClass<'l>,
1767 player: JString<'l>, target_type: JString<'l>, target_uuid: JString<'l>,
1768) {
1769 let (p, tt, tu) = (jstr!(env, player), jstr!(env, target_type), jstr!(env, target_uuid));
1770 let ev = yog_abi::YogAttackEntityEvent {
1771 player: YogStr::from_str(&p), target_type: YogStr::from_str(&tt), target_uuid: YogStr::from_str(&tu),
1772 };
1773 let srv = srv_ptr();
1774 guard("on_attack_entity", || {
1775 for (ud, f) in &handlers().attack_entity {
1776 unsafe { f(*ud, srv, &ev, 1) };
1777 }
1778 });
1779}
1780
1781#[no_mangle]
1782pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityDamage<'l>(
1783 mut env: JNIEnv<'l>, _class: JClass<'l>,
1784 entity_type: JString<'l>, uuid: JString<'l>, amount: jfloat, source: JString<'l>,
1785) {
1786 let (et, u, s) = (jstr!(env, entity_type), jstr!(env, uuid), jstr!(env, source));
1787 let ev = yog_abi::YogEntityDamageEvent {
1788 entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u),
1789 amount, source: YogStr::from_str(&s),
1790 };
1791 let srv = srv_ptr();
1792 guard("on_entity_damage", || {
1793 for (ud, f) in &handlers().entity_damage {
1794 unsafe { f(*ud, srv, &ev, 1) };
1795 }
1796 });
1797}
1798
1799#[no_mangle]
1800pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityDeath<'l>(
1801 mut env: JNIEnv<'l>, _class: JClass<'l>,
1802 entity_type: JString<'l>, uuid: JString<'l>, source: JString<'l>,
1803) {
1804 let (et, u, s) = (jstr!(env, entity_type), jstr!(env, uuid), jstr!(env, source));
1805 let ev = yog_abi::YogEntityDeathEvent {
1806 entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u), source: YogStr::from_str(&s),
1807 };
1808 let srv = srv_ptr();
1809 guard("on_entity_death", || {
1810 for (ud, f) in &handlers().entity_death {
1811 unsafe { f(*ud, srv, &ev, 1) };
1812 }
1813 });
1814}
1815
1816#[no_mangle]
1817pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnTick<'l>(
1818 _env: JNIEnv<'l>, _class: JClass<'l>,
1819) {
1820 let h = handlers();
1821 let srv = srv_ptr();
1822 guard("on_tick", || {
1823 for (ud, f) in &h.server_tick {
1824 unsafe { f(*ud, srv) };
1825 }
1826 });
1827
1828 {
1830 let mut sched = h.scheduler.lock().expect("scheduler poisoned");
1831 let mut to_fire: Vec<(*mut c_void, YogScheduledFn)> = Vec::new();
1832 let mut remaining = Vec::new();
1833 for task in sched.once_tasks.drain(..) {
1834 if task.delay_remaining == 0 {
1835 to_fire.push((task.ud, task.f));
1836 } else {
1837 remaining.push(OnceTask { delay_remaining: task.delay_remaining - 1, ..task });
1838 }
1839 }
1840 sched.once_tasks = remaining;
1841 drop(sched);
1842 for (ud, f) in to_fire {
1843 guard("schedule_once", || unsafe { f(ud, srv) });
1844 }
1845 }
1846
1847 {
1849 let mut sched = h.scheduler.lock().expect("scheduler poisoned");
1850 let mut to_fire: Vec<(*mut c_void, YogScheduledFn)> = Vec::new();
1851 for task in &mut sched.repeating_tasks {
1852 if task.ticks_left == 0 {
1853 to_fire.push((task.ud, task.f));
1854 task.ticks_left = task.period;
1855 } else {
1856 task.ticks_left -= 1;
1857 }
1858 }
1859 drop(sched);
1860 for (ud, f) in to_fire {
1861 guard("schedule_repeating", || unsafe { f(ud, srv) });
1862 }
1863 }
1864}
1865
1866#[no_mangle]
1867pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnServerStarted<'l>(
1868 _env: JNIEnv<'l>, _class: JClass<'l>,
1869) {
1870 let srv = srv_ptr();
1871 guard("on_server_started", || {
1872 for (ud, f) in &handlers().server_started {
1873 unsafe { f(*ud, srv) };
1874 }
1875 });
1876}
1877
1878#[no_mangle]
1879pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnServerStopping<'l>(
1880 _env: JNIEnv<'l>, _class: JClass<'l>,
1881) {
1882 let srv = srv_ptr();
1883 guard("on_server_stopping", || {
1884 for (ud, f) in &handlers().server_stopping {
1885 unsafe { f(*ud, srv) };
1886 }
1887 });
1888}
1889
1890#[no_mangle]
1891pub extern "system" fn Java_dev_yog_NativeBridge_nativeCommandNames<'l>(
1892 env: JNIEnv<'l>, _class: JClass<'l>,
1893) -> jstring {
1894 let names = handlers().commands.keys().cloned().collect::<Vec<_>>().join("\n");
1895 env.new_string(names).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
1896}
1897
1898#[no_mangle]
1899pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnBlockBreakPre<'l>(
1900 mut env: JNIEnv<'l>, _class: JClass<'l>,
1901 player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
1902) -> jni::sys::jboolean {
1903 let h = handlers();
1904 if h.block_break.is_empty() { return 1; }
1905 let p = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
1906 let b = match env.get_string(&block) { Ok(s) => String::from(s), Err(_) => return 1 };
1907 let ev = yog_abi::YogBlockBreakEvent {
1908 player: YogStr::from_str(&p), block: YogStr::from_str(&b),
1909 pos: YogBlockPos { x, y, z },
1910 };
1911 let srv = srv_ptr();
1912 let mut allow = true;
1913 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
1914 for (ud, f) in &h.block_break {
1915 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
1916 }
1917 })).ok();
1918 allow as jni::sys::jboolean
1919}
1920
1921#[no_mangle]
1922pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnChatPre<'l>(
1923 mut env: JNIEnv<'l>, _class: JClass<'l>,
1924 player: JString<'l>, message: JString<'l>,
1925) -> jni::sys::jboolean {
1926 let h = handlers();
1927 if h.chat.is_empty() { return 1; }
1928 let p = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
1929 let m = match env.get_string(&message) { Ok(s) => String::from(s), Err(_) => return 1 };
1930 let ev = yog_abi::YogChatEvent { player: YogStr::from_str(&p), message: YogStr::from_str(&m) };
1931 let srv = srv_ptr();
1932 let mut allow = true;
1933 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
1934 for (ud, f) in &h.chat {
1935 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
1936 }
1937 })).ok();
1938 allow as jni::sys::jboolean
1939}
1940
1941#[no_mangle]
1942pub extern "system" fn Java_dev_yog_NativeBridge_nativeRecipeJsons<'l>(
1943 env: JNIEnv<'l>, _class: JClass<'l>,
1944) -> jstring {
1945 let s = handlers().recipes.iter()
1946 .map(|(ns, name, json)| format!("{}\t{}\t{}", ns, name, json))
1947 .collect::<Vec<_>>().join("\n");
1948 env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
1949}
1950
1951#[no_mangle]
1952pub extern "system" fn Java_dev_yog_NativeBridge_nativeTypedCommandSchemas<'l>(
1953 env: JNIEnv<'l>, _class: JClass<'l>,
1954) -> jstring {
1955 let s = handlers().typed_schemas.iter()
1956 .map(|(name, schema)| format!("{}\t{}", name, schema))
1957 .collect::<Vec<_>>().join("\n");
1958 env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
1959}
1960
1961#[no_mangle]
1962pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnCommand<'l>(
1963 mut env: JNIEnv<'l>, _class: JClass<'l>,
1964 name: JString<'l>, args: JString<'l>, source: JString<'l>, uuid: JString<'l>,
1965) -> jstring {
1966 let (n, a, s, u) = (
1967 env.get_string(&name).map(String::from).unwrap_or_default(),
1968 env.get_string(&args).map(String::from).unwrap_or_default(),
1969 env.get_string(&source).map(String::from).unwrap_or_default(),
1970 env.get_string(&uuid).map(String::from).unwrap_or_default(),
1971 );
1972 let ev = yog_abi::YogCommandEvent {
1973 name: YogStr::from_str(&n), args: YogStr::from_str(&a),
1974 source: YogStr::from_str(&s), uuid: YogStr::from_str(&u),
1975 };
1976 let h = handlers();
1977 let srv = srv_ptr();
1978 let reply = std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
1979 if let Some((ud, f)) = h.commands.get(&n) {
1980 let mut buf = [0u8; 4096];
1981 let mut reply_len: u32 = 0;
1982 unsafe { f(*ud, srv, &ev, buf.as_mut_ptr(), buf.len() as u32, &mut reply_len) };
1983 String::from_utf8_lossy(&buf[..reply_len as usize]).into_owned()
1984 } else {
1985 String::new()
1986 }
1987 }))
1988 .unwrap_or_else(|_| { yog_logging::error!("a mod panicked handling command `{}`", n); String::new() });
1989
1990 env.new_string(reply).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
1991}
1992
1993#[no_mangle]
1994pub extern "system" fn Java_dev_yog_NativeBridge_nativeItemDefs<'l>(
1995 env: JNIEnv<'l>, _class: JClass<'l>,
1996) -> jstring {
1997 let s = handlers().items.iter().map(|d| {
1998 let mut parts = vec![d.id.clone()];
1999 parts.push(format!("max_stack={}", d.max_stack));
2000 if let Some(n) = &d.name { parts.push(format!("name={n}")); }
2001 if let Some(t) = &d.tooltip { parts.push(format!("tooltip={t}")); }
2002 if d.max_damage > 0 { parts.push(format!("max_damage={}", d.max_damage)); }
2003 if d.fire_resistant { parts.push("fire_resistant=1".into()); }
2004 if d.fuel_ticks > 0 { parts.push(format!("fuel_ticks={}", d.fuel_ticks)); }
2005 if let Some(f) = &d.food {
2006 parts.push(format!("food={}:{}:{}", f.nutrition, f.saturation, if f.can_always_eat { 1 } else { 0 }));
2007 }
2008 parts.join("\t")
2009 }).collect::<Vec<_>>().join("\n");
2010 env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2011}
2012
2013#[no_mangle]
2014pub extern "system" fn Java_dev_yog_NativeBridge_nativeBlockDefs<'l>(
2015 env: JNIEnv<'l>, _class: JClass<'l>,
2016) -> jstring {
2017 let s = handlers().blocks.iter().map(|d| {
2018 let mut parts = vec![d.id.clone()];
2019 parts.push(format!("hardness={}", d.hardness));
2020 parts.push(format!("resistance={}", d.resistance));
2021 if let Some(n) = &d.name { parts.push(format!("name={n}")); }
2022 if let Some(sh) = d.shape {
2023 parts.push(format!("shape={}:{}:{}:{}:{}:{}", sh[0], sh[1], sh[2], sh[3], sh[4], sh[5]));
2024 }
2025 if d.light_level > 0 { parts.push(format!("light={}", d.light_level)); }
2026 if let Some(snd) = &d.sound { parts.push(format!("sound={snd}")); }
2027 if d.requires_tool { parts.push("requires_tool=1".into()); }
2028 if d.no_collision { parts.push("no_collision=1".into()); }
2029 if d.slipperiness > 0.0 { parts.push(format!("slipperiness={}", d.slipperiness)); }
2030 parts.join("\t")
2031 }).collect::<Vec<_>>().join("\n");
2032 env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2033}
2034
2035#[no_mangle]
2036pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPacket<'l>(
2037 mut env: JNIEnv<'l>, _class: JClass<'l>,
2038 channel: JString<'l>, player: JString<'l>, payload: JByteArray<'l>,
2039) {
2040 let ch = env.get_string(&channel).map(String::from).unwrap_or_default();
2041 let pl = env.get_string(&player).map(String::from).unwrap_or_default();
2042 let data = env.convert_byte_array(&payload).unwrap_or_default();
2043 let ev = yog_abi::YogPacketEvent {
2044 channel: YogStr::from_str(&ch), player: YogStr::from_str(&pl),
2045 payload: data.as_ptr(), payload_len: data.len() as u32,
2046 };
2047 let h = handlers();
2048 let srv = srv_ptr();
2049 guard("on_packet", || {
2050 if let Some((ud, f)) = h.packets.get(&ch) {
2051 unsafe { f(*ud, srv, &ev) };
2052 }
2053 });
2054}
2055
2056#[no_mangle]
2057pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnClientPacket<'l>(
2058 mut env: JNIEnv<'l>, _class: JClass<'l>,
2059 channel: JString<'l>, payload: JByteArray<'l>,
2060) {
2061 let ch = env.get_string(&channel).map(String::from).unwrap_or_default();
2062 let data = env.convert_byte_array(&payload).unwrap_or_default();
2063 let ev = yog_abi::YogPacketEvent {
2064 channel: YogStr::from_str(&ch), player: YogStr::EMPTY,
2065 payload: data.as_ptr(), payload_len: data.len() as u32,
2066 };
2067 let h = handlers();
2068 let srv = srv_ptr();
2069 guard("on_client_packet", || {
2070 if let Some((ud, f)) = h.client_packets.get(&ch) {
2071 unsafe { f(*ud, srv, &ev) };
2072 }
2073 });
2074}
2075
2076#[no_mangle]
2077pub extern "system" fn Java_dev_yog_NativeBridge_nativePacketChannels<'l>(
2078 env: JNIEnv<'l>, _class: JClass<'l>,
2079) -> jstring {
2080 let s = handlers().packets.keys().cloned().collect::<Vec<_>>().join("\n");
2081 env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2082}
2083
2084#[no_mangle]
2085pub extern "system" fn Java_dev_yog_NativeBridge_nativeClientPacketChannels<'l>(
2086 env: JNIEnv<'l>, _class: JClass<'l>,
2087) -> jstring {
2088 let s = handlers().client_packets.keys().cloned().collect::<Vec<_>>().join("\n");
2089 env.new_string(s).map(|s| s.into_raw()).unwrap_or(std::ptr::null_mut())
2090}
2091
2092#[no_mangle]
2093pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntitySpawn<'l>(
2094 mut env: JNIEnv<'l>, _class: JClass<'l>,
2095 entity_type: JString<'l>, uuid: JString<'l>, dimension: JString<'l>,
2096) {
2097 let h = handlers();
2098 if h.entity_spawn.is_empty() { return; }
2099 let (et, u, d) = (jstr!(env, entity_type), jstr!(env, uuid), jstr!(env, dimension));
2100 let ev = yog_abi::YogEntitySpawnEvent {
2101 entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u),
2102 dimension: YogStr::from_str(&d),
2103 };
2104 let srv = srv_ptr();
2105 guard("on_entity_spawn", || {
2106 for (ud, f) in &h.entity_spawn {
2107 unsafe { f(*ud, srv, &ev, 1) };
2108 }
2109 });
2110}
2111
2112#[no_mangle]
2113pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntitySpawnPre<'l>(
2114 mut env: JNIEnv<'l>, _class: JClass<'l>,
2115 entity_type: JString<'l>, uuid: JString<'l>, dimension: JString<'l>,
2116) -> jni::sys::jboolean {
2117 let h = handlers();
2118 if h.entity_spawn.is_empty() { return 1; }
2119 let et = match env.get_string(&entity_type) { Ok(s) => String::from(s), Err(_) => return 1 };
2120 let u = match env.get_string(&uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2121 let d = match env.get_string(&dimension) { Ok(s) => String::from(s), Err(_) => return 1 };
2122 let ev = yog_abi::YogEntitySpawnEvent {
2123 entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u),
2124 dimension: YogStr::from_str(&d),
2125 };
2126 let srv = srv_ptr();
2127 let mut allow = true;
2128 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2129 for (ud, f) in &h.entity_spawn {
2130 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2131 }
2132 })).ok();
2133 allow as jni::sys::jboolean
2134}
2135
2136#[no_mangle]
2137pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityDamagePre<'l>(
2138 mut env: JNIEnv<'l>, _class: JClass<'l>,
2139 entity_type: JString<'l>, uuid: JString<'l>, amount: jfloat, source: JString<'l>,
2140) -> jni::sys::jboolean {
2141 let h = handlers();
2142 if h.entity_damage.is_empty() { return 1; }
2143 let et = match env.get_string(&entity_type) { Ok(s) => String::from(s), Err(_) => return 1 };
2144 let u = match env.get_string(&uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2145 let s = match env.get_string(&source) { Ok(s) => String::from(s), Err(_) => return 1 };
2146 let ev = yog_abi::YogEntityDamageEvent {
2147 entity_type: YogStr::from_str(&et), uuid: YogStr::from_str(&u),
2148 amount, source: YogStr::from_str(&s),
2149 };
2150 let srv = srv_ptr();
2151 let mut allow = true;
2152 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2153 for (ud, f) in &h.entity_damage {
2154 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2155 }
2156 })).ok();
2157 allow as jni::sys::jboolean
2158}
2159
2160#[no_mangle]
2161pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlaceBlockPre<'l>(
2162 mut env: JNIEnv<'l>, _class: JClass<'l>,
2163 player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
2164) -> jni::sys::jboolean {
2165 let h = handlers();
2166 if h.player_place_block.is_empty() { return 1; }
2167 let p = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
2168 let b = match env.get_string(&block) { Ok(s) => String::from(s), Err(_) => return 1 };
2169 let ev = YogPlaceBlockEvent {
2170 player: YogStr::from_str(&p), block: YogStr::from_str(&b),
2171 pos: YogBlockPos { x, y, z },
2172 };
2173 let srv = srv_ptr();
2174 let mut allow = true;
2175 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2176 for (ud, f) in &h.player_place_block {
2177 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2178 }
2179 })).ok();
2180 allow as jni::sys::jboolean
2181}
2182
2183#[no_mangle]
2184pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlaceBlock<'l>(
2185 mut env: JNIEnv<'l>, _class: JClass<'l>,
2186 player: JString<'l>, block: JString<'l>, x: jint, y: jint, z: jint,
2187) {
2188 let h = handlers();
2189 if h.player_place_block.is_empty() { return; }
2190 let (p, b) = (jstr!(env, player), jstr!(env, block));
2191 let ev = YogPlaceBlockEvent {
2192 player: YogStr::from_str(&p), block: YogStr::from_str(&b),
2193 pos: YogBlockPos { x, y, z },
2194 };
2195 let srv = srv_ptr();
2196 guard("on_player_place_block", || {
2197 for (ud, f) in &h.player_place_block {
2198 unsafe { f(*ud, srv, &ev, 1) };
2199 }
2200 });
2201}
2202
2203#[no_mangle]
2204pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerDeathPre<'l>(
2205 mut env: JNIEnv<'l>, _class: JClass<'l>,
2206 player: JString<'l>, uuid: JString<'l>, source: JString<'l>,
2207) -> jni::sys::jboolean {
2208 let h = handlers();
2209 if h.player_death.is_empty() { return 1; }
2210 let p = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
2211 let u = match env.get_string(&uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2212 let s = match env.get_string(&source) { Ok(s) => String::from(s), Err(_) => return 1 };
2213 let ev = YogPlayerDeathEvent {
2214 player: YogStr::from_str(&p), uuid: YogStr::from_str(&u), source: YogStr::from_str(&s),
2215 };
2216 let srv = srv_ptr();
2217 let mut allow = true;
2218 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2219 for (ud, f) in &h.player_death {
2220 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2221 }
2222 })).ok();
2223 allow as jni::sys::jboolean
2224}
2225
2226#[no_mangle]
2227pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerDeath<'l>(
2228 mut env: JNIEnv<'l>, _class: JClass<'l>,
2229 player: JString<'l>, uuid: JString<'l>, source: JString<'l>,
2230) {
2231 let h = handlers();
2232 if h.player_death.is_empty() { return; }
2233 let (p, u, s) = (jstr!(env, player), jstr!(env, uuid), jstr!(env, source));
2234 let ev = YogPlayerDeathEvent {
2235 player: YogStr::from_str(&p), uuid: YogStr::from_str(&u), source: YogStr::from_str(&s),
2236 };
2237 let srv = srv_ptr();
2238 guard("on_player_death", || {
2239 for (ud, f) in &h.player_death {
2240 unsafe { f(*ud, srv, &ev, 1) };
2241 }
2242 });
2243}
2244
2245#[no_mangle]
2246pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerRespawn<'l>(
2247 mut env: JNIEnv<'l>, _class: JClass<'l>,
2248 player: JString<'l>, uuid: JString<'l>, at_anchor: jni::sys::jboolean,
2249) {
2250 let h = handlers();
2251 if h.player_respawn.is_empty() { return; }
2252 let (p, u) = (jstr!(env, player), jstr!(env, uuid));
2253 let ev = YogPlayerRespawnEvent {
2254 player: YogStr::from_str(&p), uuid: YogStr::from_str(&u), at_anchor: at_anchor != 0,
2255 };
2256 let srv = srv_ptr();
2257 guard("on_player_respawn", || {
2258 for (ud, f) in &h.player_respawn {
2259 unsafe { f(*ud, srv, &ev, 1) };
2260 }
2261 });
2262}
2263
2264#[no_mangle]
2265pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnAdvancement<'l>(
2266 mut env: JNIEnv<'l>, _class: JClass<'l>,
2267 player: JString<'l>, uuid: JString<'l>, advancement: JString<'l>,
2268) {
2269 let h = handlers();
2270 if h.advancement.is_empty() { return; }
2271 let (p, u, a) = (jstr!(env, player), jstr!(env, uuid), jstr!(env, advancement));
2272 let ev = YogAdvancementEvent {
2273 player: YogStr::from_str(&p), uuid: YogStr::from_str(&u), advancement: YogStr::from_str(&a),
2274 };
2275 let srv = srv_ptr();
2276 guard("on_advancement", || {
2277 for (ud, f) in &h.advancement {
2278 unsafe { f(*ud, srv, &ev, 1) };
2279 }
2280 });
2281}
2282
2283#[no_mangle]
2284pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityInteractPre<'l>(
2285 mut env: JNIEnv<'l>, _class: JClass<'l>,
2286 player: JString<'l>, player_uuid: JString<'l>,
2287 entity_type: JString<'l>, entity_uuid: JString<'l>, hand: JString<'l>,
2288) -> jni::sys::jboolean {
2289 let h = handlers();
2290 if h.entity_interact.is_empty() { return 1; }
2291 let p = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
2292 let pu = match env.get_string(&player_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2293 let et = match env.get_string(&entity_type) { Ok(s) => String::from(s), Err(_) => return 1 };
2294 let eu = match env.get_string(&entity_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2295 let ha = match env.get_string(&hand) { Ok(s) => String::from(s), Err(_) => return 1 };
2296 let ev = YogEntityInteractEvent {
2297 player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2298 entity_type: YogStr::from_str(&et), entity_uuid: YogStr::from_str(&eu),
2299 hand: YogStr::from_str(&ha),
2300 };
2301 let srv = srv_ptr();
2302 let mut allow = true;
2303 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2304 for (ud, f) in &h.entity_interact {
2305 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2306 }
2307 })).ok();
2308 allow as jni::sys::jboolean
2309}
2310
2311#[no_mangle]
2312pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnEntityInteract<'l>(
2313 mut env: JNIEnv<'l>, _class: JClass<'l>,
2314 player: JString<'l>, player_uuid: JString<'l>,
2315 entity_type: JString<'l>, entity_uuid: JString<'l>, hand: JString<'l>,
2316) {
2317 let h = handlers();
2318 if h.entity_interact.is_empty() { return; }
2319 let (p, pu) = (jstr!(env, player), jstr!(env, player_uuid));
2320 let (et, eu, ha) = (jstr!(env, entity_type), jstr!(env, entity_uuid), jstr!(env, hand));
2321 let ev = YogEntityInteractEvent {
2322 player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2323 entity_type: YogStr::from_str(&et), entity_uuid: YogStr::from_str(&eu),
2324 hand: YogStr::from_str(&ha),
2325 };
2326 let srv = srv_ptr();
2327 guard("on_entity_interact", || {
2328 for (ud, f) in &h.entity_interact {
2329 unsafe { f(*ud, srv, &ev, 1) };
2330 }
2331 });
2332}
2333
2334#[no_mangle]
2335pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnItemCraft<'l>(
2336 mut env: JNIEnv<'l>, _class: JClass<'l>,
2337 player: JString<'l>, player_uuid: JString<'l>,
2338 result_item: JString<'l>, result_count: jint,
2339) {
2340 let h = handlers();
2341 if h.item_craft.is_empty() { return; }
2342 let (p, pu, ri) = (jstr!(env, player), jstr!(env, player_uuid), jstr!(env, result_item));
2343 let ev = YogCraftEvent {
2344 player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2345 result_item: YogStr::from_str(&ri), result_count: result_count as u32,
2346 };
2347 let srv = srv_ptr();
2348 guard("on_item_craft", || {
2349 for (ud, f) in &h.item_craft {
2350 unsafe { f(*ud, srv, &ev, 1) };
2351 }
2352 });
2353}
2354
2355#[no_mangle]
2356pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnExplosionPre<'l>(
2357 mut env: JNIEnv<'l>, _class: JClass<'l>,
2358 dimension: JString<'l>, x: jdouble, y: jdouble, z: jdouble,
2359 power: jfloat, cause_uuid: JString<'l>,
2360) -> jni::sys::jboolean {
2361 let h = handlers();
2362 if h.explosion.is_empty() { return 1; }
2363 let d = match env.get_string(&dimension) { Ok(s) => String::from(s), Err(_) => return 1 };
2364 let cu = match env.get_string(&cause_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2365 let ev = YogExplosionEvent {
2366 dimension: YogStr::from_str(&d), x, y, z, power, cause_uuid: YogStr::from_str(&cu),
2367 };
2368 let srv = srv_ptr();
2369 let mut allow = true;
2370 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2371 for (ud, f) in &h.explosion {
2372 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2373 }
2374 })).ok();
2375 allow as jni::sys::jboolean
2376}
2377
2378#[no_mangle]
2379pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnExplosion<'l>(
2380 mut env: JNIEnv<'l>, _class: JClass<'l>,
2381 dimension: JString<'l>, x: jdouble, y: jdouble, z: jdouble,
2382 power: jfloat, cause_uuid: JString<'l>,
2383) {
2384 let h = handlers();
2385 if h.explosion.is_empty() { return; }
2386 let (d, cu) = (jstr!(env, dimension), jstr!(env, cause_uuid));
2387 let ev = YogExplosionEvent {
2388 dimension: YogStr::from_str(&d), x, y, z, power, cause_uuid: YogStr::from_str(&cu),
2389 };
2390 let srv = srv_ptr();
2391 guard("on_explosion", || {
2392 for (ud, f) in &h.explosion {
2393 unsafe { f(*ud, srv, &ev, 1) };
2394 }
2395 });
2396}
2397
2398#[no_mangle]
2401pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnItemPickupPre<'l>(
2402 mut env: JNIEnv<'l>, _class: JClass<'l>,
2403 player: JString<'l>, player_uuid: JString<'l>,
2404 item_id: JString<'l>, item_count: jint, entity_uuid: JString<'l>,
2405) -> jni::sys::jboolean {
2406 let h = handlers();
2407 if h.item_pickup.is_empty() { return 1; }
2408 let p = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
2409 let pu = match env.get_string(&player_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2410 let ii = match env.get_string(&item_id) { Ok(s) => String::from(s), Err(_) => return 1 };
2411 let eu = match env.get_string(&entity_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2412 let ev = YogItemPickupEvent {
2413 player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2414 item_id: YogStr::from_str(&ii), item_count: item_count as u32,
2415 entity_uuid: YogStr::from_str(&eu),
2416 };
2417 let srv = srv_ptr();
2418 let mut allow = true;
2419 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2420 for (ud, f) in &h.item_pickup {
2421 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2422 }
2423 })).ok();
2424 allow as jni::sys::jboolean
2425}
2426
2427#[no_mangle]
2428pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnItemPickup<'l>(
2429 mut env: JNIEnv<'l>, _class: JClass<'l>,
2430 player: JString<'l>, player_uuid: JString<'l>,
2431 item_id: JString<'l>, item_count: jint, entity_uuid: JString<'l>,
2432) {
2433 let h = handlers();
2434 if h.item_pickup.is_empty() { return; }
2435 let (p, pu) = (jstr!(env, player), jstr!(env, player_uuid));
2436 let (ii, eu) = (jstr!(env, item_id), jstr!(env, entity_uuid));
2437 let ev = YogItemPickupEvent {
2438 player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2439 item_id: YogStr::from_str(&ii), item_count: item_count as u32,
2440 entity_uuid: YogStr::from_str(&eu),
2441 };
2442 let srv = srv_ptr();
2443 guard("on_item_pickup", || {
2444 for (ud, f) in &h.item_pickup {
2445 unsafe { f(*ud, srv, &ev, 1) };
2446 }
2447 });
2448}
2449
2450#[no_mangle]
2451pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnPlayerMove<'l>(
2452 mut env: JNIEnv<'l>, _class: JClass<'l>,
2453 player: JString<'l>, player_uuid: JString<'l>,
2454 x: jdouble, y: jdouble, z: jdouble, yaw: jfloat, pitch: jfloat,
2455) {
2456 let h = handlers();
2457 if h.player_move.is_empty() { return; }
2458 let (p, pu) = (jstr!(env, player), jstr!(env, player_uuid));
2459 let ev = YogPlayerMoveEvent {
2460 player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2461 x, y, z, yaw, pitch,
2462 };
2463 let srv = srv_ptr();
2464 guard("on_player_move", || {
2465 for (ud, f) in &h.player_move {
2466 unsafe { f(*ud, srv, &ev, 1) };
2467 }
2468 });
2469}
2470
2471#[no_mangle]
2472pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnContainerOpenPre<'l>(
2473 mut env: JNIEnv<'l>, _class: JClass<'l>,
2474 player: JString<'l>, player_uuid: JString<'l>,
2475) -> jni::sys::jboolean {
2476 let h = handlers();
2477 if h.container_open.is_empty() { return 1; }
2478 let p = match env.get_string(&player) { Ok(s) => String::from(s), Err(_) => return 1 };
2479 let pu = match env.get_string(&player_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2480 let ev = YogContainerOpenEvent {
2481 player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2482 container_type: YogStr::EMPTY,
2483 };
2484 let srv = srv_ptr();
2485 let mut allow = true;
2486 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2487 for (ud, f) in &h.container_open {
2488 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2489 }
2490 })).ok();
2491 allow as jni::sys::jboolean
2492}
2493
2494#[no_mangle]
2495pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnContainerOpen<'l>(
2496 mut env: JNIEnv<'l>, _class: JClass<'l>,
2497 player: JString<'l>, player_uuid: JString<'l>, container_type: JString<'l>,
2498) {
2499 let h = handlers();
2500 if h.container_open.is_empty() { return; }
2501 let (p, pu, ct) = (jstr!(env, player), jstr!(env, player_uuid), jstr!(env, container_type));
2502 let ev = YogContainerOpenEvent {
2503 player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2504 container_type: YogStr::from_str(&ct),
2505 };
2506 let srv = srv_ptr();
2507 guard("on_container_open", || {
2508 for (ud, f) in &h.container_open {
2509 unsafe { f(*ud, srv, &ev, 1) };
2510 }
2511 });
2512}
2513
2514#[no_mangle]
2515pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnContainerClose<'l>(
2516 mut env: JNIEnv<'l>, _class: JClass<'l>,
2517 player: JString<'l>, player_uuid: JString<'l>,
2518) {
2519 let h = handlers();
2520 if h.container_close.is_empty() { return; }
2521 let (p, pu) = (jstr!(env, player), jstr!(env, player_uuid));
2522 let ev = YogContainerCloseEvent {
2523 player: YogStr::from_str(&p), player_uuid: YogStr::from_str(&pu),
2524 };
2525 let srv = srv_ptr();
2526 guard("on_container_close", || {
2527 for (ud, f) in &h.container_close {
2528 unsafe { f(*ud, srv, &ev, 1) };
2529 }
2530 });
2531}
2532
2533#[no_mangle]
2534pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnProjectileHitPre<'l>(
2535 mut env: JNIEnv<'l>, _class: JClass<'l>,
2536 projectile_type: JString<'l>, projectile_uuid: JString<'l>, shooter_uuid: JString<'l>,
2537 hit_type: JString<'l>, hit_entity_uuid: JString<'l>,
2538 x: jdouble, y: jdouble, z: jdouble, dimension: JString<'l>,
2539) -> jni::sys::jboolean {
2540 let h = handlers();
2541 if h.projectile_hit.is_empty() { return 1; }
2542 let pt = match env.get_string(&projectile_type) { Ok(s) => String::from(s), Err(_) => return 1 };
2543 let pu = match env.get_string(&projectile_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2544 let su = match env.get_string(&shooter_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2545 let ht = match env.get_string(&hit_type) { Ok(s) => String::from(s), Err(_) => return 1 };
2546 let heu = match env.get_string(&hit_entity_uuid) { Ok(s) => String::from(s), Err(_) => return 1 };
2547 let dim = match env.get_string(&dimension) { Ok(s) => String::from(s), Err(_) => return 1 };
2548 let ev = YogProjectileHitEvent {
2549 projectile_type: YogStr::from_str(&pt), projectile_uuid: YogStr::from_str(&pu),
2550 shooter_uuid: YogStr::from_str(&su), hit_type: YogStr::from_str(&ht),
2551 hit_entity_uuid: YogStr::from_str(&heu), x, y, z, dimension: YogStr::from_str(&dim),
2552 };
2553 let srv = srv_ptr();
2554 let mut allow = true;
2555 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2556 for (ud, f) in &h.projectile_hit {
2557 if !unsafe { f(*ud, srv, &ev, 0) } { allow = false; break; }
2558 }
2559 })).ok();
2560 allow as jni::sys::jboolean
2561}
2562
2563#[no_mangle]
2564pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnProjectileHit<'l>(
2565 mut env: JNIEnv<'l>, _class: JClass<'l>,
2566 projectile_type: JString<'l>, projectile_uuid: JString<'l>, shooter_uuid: JString<'l>,
2567 hit_type: JString<'l>, hit_entity_uuid: JString<'l>,
2568 x: jdouble, y: jdouble, z: jdouble, dimension: JString<'l>,
2569) {
2570 let h = handlers();
2571 if h.projectile_hit.is_empty() { return; }
2572 let (pt, pu) = (jstr!(env, projectile_type), jstr!(env, projectile_uuid));
2573 let (su, ht) = (jstr!(env, shooter_uuid), jstr!(env, hit_type));
2574 let (heu, dim) = (jstr!(env, hit_entity_uuid), jstr!(env, dimension));
2575 let ev = YogProjectileHitEvent {
2576 projectile_type: YogStr::from_str(&pt), projectile_uuid: YogStr::from_str(&pu),
2577 shooter_uuid: YogStr::from_str(&su), hit_type: YogStr::from_str(&ht),
2578 hit_entity_uuid: YogStr::from_str(&heu), x, y, z, dimension: YogStr::from_str(&dim),
2579 };
2580 let srv = srv_ptr();
2581 guard("on_projectile_hit", || {
2582 for (ud, f) in &h.projectile_hit {
2583 unsafe { f(*ud, srv, &ev, 1) };
2584 }
2585 });
2586}
2587
2588#[no_mangle]
2591pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnClientTick<'l>(
2592 _env: JNIEnv<'l>, _class: JClass<'l>,
2593) {
2594 let h = handlers();
2595 if h.client_tick.is_empty() { return; }
2596 guard("on_client_tick", || {
2597 for (ud, f) in &h.client_tick {
2598 unsafe { f(*ud) };
2599 }
2600 });
2601}
2602
2603#[no_mangle]
2604pub extern "system" fn Java_dev_yog_NativeBridge_nativeGlInit<'l>(
2605 _env: JNIEnv<'l>, _class: JClass<'l>,
2606) {
2607 if GL.get().is_some() { return; }
2608 let mut raw_get_binary: usize = 0;
2609 let mut raw_prog_binary: usize = 0;
2610 let gl = unsafe {
2611 glow::Context::from_loader_function(|sym| {
2612 let Some(mut env) = get_env() else { return std::ptr::null() };
2613 let jsym = match env.new_string(sym) {
2614 Ok(s) => s,
2615 Err(_) => return std::ptr::null(),
2616 };
2617 let jsym_obj: JObject = jsym.into();
2618 let val = env.call_static_method(
2619 "dev/yog/NativeBridge",
2620 "glProcAddress",
2621 "(Ljava/lang/String;)J",
2622 &[JValue::Object(&jsym_obj)],
2623 );
2624 let ptr = match val.and_then(|v| v.j()) {
2625 Ok(p) if p != 0 => p as usize as *const _,
2626 _ => std::ptr::null(),
2627 };
2628 match sym {
2630 "glGetProgramBinary" => raw_get_binary = ptr as usize,
2631 "glProgramBinary" => raw_prog_binary = ptr as usize,
2632 _ => {}
2633 }
2634 ptr
2635 })
2636 };
2637 let _ = GL.set(GlCtx(gl));
2638 let _ = GL_GET_PROGRAM_BINARY.set(if raw_get_binary != 0 { Some(raw_get_binary) } else { None });
2639 let _ = GL_PROGRAM_BINARY.set(if raw_prog_binary != 0 { Some(raw_prog_binary) } else { None });
2640 if let Some(mut env) = get_env() {
2643 if let Ok(jsym) = env.new_string("glGetProgramiv") {
2644 let jsym_obj: JObject = jsym.into();
2645 if let Ok(jv) = env.call_static_method(
2646 "dev/yog/NativeBridge", "glProcAddress", "(Ljava/lang/String;)J",
2647 &[JValue::Object(&jsym_obj)],
2648 ) {
2649 if let Ok(ptr) = jv.j() {
2650 let _ = GL_GET_PROGRAM_IV.set(if ptr != 0 { Some(ptr as usize) } else { None });
2651 }
2652 }
2653 }
2654 }
2655}
2656
2657#[no_mangle]
2658pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnHudRender<'l>(
2659 _env: JNIEnv<'l>, _class: JClass<'l>,
2660 delta_tick: jfloat,
2661 screen_w: jint,
2662 screen_h: jint,
2663 scale_factor: jfloat,
2664 player_x: jfloat, player_y: jfloat, player_z: jfloat,
2665) {
2666 let h = handlers();
2667 if h.hud_render.is_empty() { return; }
2668 let mut gfx = GFX_FN_TABLE;
2669 gfx.screen_w = screen_w;
2670 gfx.screen_h = screen_h;
2671 gfx.delta_tick = delta_tick;
2672 gfx.scale_factor = scale_factor;
2673 gfx.player_pos = [player_x, player_y, player_z];
2674 guard("on_hud_render", || {
2675 for (ud, f) in &h.hud_render {
2676 unsafe { f(*ud, &gfx) };
2677 }
2678 });
2679}
2680
2681#[no_mangle]
2682pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnWorldRender<'l>(
2683 env: JNIEnv<'l>, _class: JClass<'l>,
2684 delta_tick: jfloat,
2685 screen_w: jint,
2686 screen_h: jint,
2687 scale_factor: jfloat,
2688 view_proj_arr: JFloatArray<'l>,
2689 cam_x: jfloat, cam_y: jfloat, cam_z: jfloat,
2690 player_x: jfloat, player_y: jfloat, player_z: jfloat,
2691) {
2692 let h = handlers();
2693 if h.world_render.is_empty() { return; }
2694 let mut view_proj = [0f32; 16];
2695 if env.get_float_array_region(&view_proj_arr, 0, &mut view_proj).is_err() { return; }
2696 let mut gfx = GFX_FN_TABLE;
2697 gfx.screen_w = screen_w;
2698 gfx.screen_h = screen_h;
2699 gfx.delta_tick = delta_tick;
2700 gfx.scale_factor = scale_factor;
2701 gfx.view_proj = view_proj;
2702 gfx.camera_pos = [cam_x, cam_y, cam_z];
2703 gfx.player_pos = [player_x, player_y, player_z];
2704 guard("on_world_render", || {
2705 for (ud, f) in &h.world_render {
2706 unsafe { f(*ud, &gfx) };
2707 }
2708 });
2709}
2710
2711#[no_mangle]
2712pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnKeyPress<'l>(
2713 _env: JNIEnv<'l>, _class: JClass<'l>,
2714 key_code: jint, scan_code: jint, action: jint, modifiers: jint,
2715) -> jni::sys::jboolean {
2716 let h = handlers();
2717 if h.key_press.is_empty() { return 1; }
2718 let ev = YogKeyPressEvent { key_code, scan_code, action, modifiers };
2719 let mut allow = true;
2720 std::panic::catch_unwind(std::panic::AssertUnwindSafe(|| {
2721 for (ud, f) in &h.key_press {
2722 if !unsafe { f(*ud, &ev) } { allow = false; break; }
2723 }
2724 })).ok();
2725 allow as jni::sys::jboolean
2726}
2727
2728#[no_mangle]
2729pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnScreenOpen<'l>(
2730 mut env: JNIEnv<'l>, _class: JClass<'l>,
2731 screen_class: JString<'l>,
2732) {
2733 let h = handlers();
2734 if h.screen_open.is_empty() { return; }
2735 let sc = match env.get_string(&screen_class) { Ok(s) => String::from(s), Err(_) => return };
2736 guard("on_screen_open", || {
2737 for (ud, f) in &h.screen_open {
2738 unsafe { f(*ud, YogStr::from_str(&sc)) };
2739 }
2740 });
2741}
2742
2743#[no_mangle]
2744pub extern "system" fn Java_dev_yog_NativeBridge_nativeOnScreenClose<'l>(
2745 mut env: JNIEnv<'l>, _class: JClass<'l>,
2746 screen_class: JString<'l>,
2747) {
2748 let h = handlers();
2749 if h.screen_close.is_empty() { return; }
2750 let sc = match env.get_string(&screen_class) { Ok(s) => String::from(s), Err(_) => return };
2751 guard("on_screen_close", || {
2752 for (ud, f) in &h.screen_close {
2753 unsafe { f(*ud, YogStr::from_str(&sc)) };
2754 }
2755 });
2756}