Hi
This commit is contained in:
parent
a6b2c6bed8
commit
d897141dd5
42 changed files with 2419 additions and 2 deletions
2
.cargo/config.toml
Normal file
2
.cargo/config.toml
Normal file
|
@ -0,0 +1,2 @@
|
|||
[build]
|
||||
target = "armv7-linux-androideabi"
|
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
|
@ -0,0 +1 @@
|
|||
target/
|
639
Cargo.lock
generated
Normal file
639
Cargo.lock
generated
Normal file
|
@ -0,0 +1,639 @@
|
|||
# This file is automatically @generated by Cargo.
|
||||
# It is not intended for manual editing.
|
||||
version = 4
|
||||
|
||||
[[package]]
|
||||
name = "aho-corasick"
|
||||
version = "1.1.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8e60d3430d3a69478ad0993f19238d2df97c507009a52b3c10addcd7f6bcb916"
|
||||
dependencies = [
|
||||
"memchr",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "android_log-sys"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "85965b6739a430150bdd138e2374a98af0c3ee0d030b3bb7fc3bddff58d0102e"
|
||||
|
||||
[[package]]
|
||||
name = "bitflags"
|
||||
version = "2.9.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5c8214115b7bf84099f1309324e63141d4c5d7cc26862f97a0a857dbefe165bd"
|
||||
|
||||
[[package]]
|
||||
name = "bytes"
|
||||
version = "1.10.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d71b6127be86fdcfddb610f7182ac57211d4b18a3e9c82eb2d17662f2227ad6a"
|
||||
|
||||
[[package]]
|
||||
name = "cesu8"
|
||||
version = "1.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "6d43a04d8753f35258c91f8ec639f792891f748a1edbd759cf1dcea3382ad83c"
|
||||
|
||||
[[package]]
|
||||
name = "cfg-if"
|
||||
version = "1.0.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
|
||||
|
||||
[[package]]
|
||||
name = "combine"
|
||||
version = "4.6.7"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ba5a308b75df32fe02788e748662718f03fde005016435c444eea572398219fd"
|
||||
dependencies = [
|
||||
"bytes",
|
||||
"memchr",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "getrandom"
|
||||
version = "0.3.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "73fea8450eea4bac3940448fb7ae50d91f034f941199fcd9d909a5a07aa455f0"
|
||||
dependencies = [
|
||||
"cfg-if",
|
||||
"libc",
|
||||
"r-efi",
|
||||
"wasi",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "jni"
|
||||
version = "0.21.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "1a87aa2bb7d2af34197c04845522473242e1aa17c12f4935d5856491a7fb8c97"
|
||||
dependencies = [
|
||||
"cesu8",
|
||||
"cfg-if",
|
||||
"combine",
|
||||
"jni-sys",
|
||||
"log",
|
||||
"thiserror",
|
||||
"walkdir",
|
||||
"windows-sys 0.45.0",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "jni-sys"
|
||||
version = "0.3.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8eaf4bc02d17cbdd7ff4c7438cafcdf7fb9a4613313ad11b4f8fefe7d3fa0130"
|
||||
|
||||
[[package]]
|
||||
name = "lazy_static"
|
||||
version = "1.5.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe"
|
||||
|
||||
[[package]]
|
||||
name = "libc"
|
||||
version = "0.2.171"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c19937216e9d3aa9956d9bb8dfc0b0c8beb6058fc4f7a4dc4d850edf86a237d6"
|
||||
|
||||
[[package]]
|
||||
name = "libserver"
|
||||
version = "0.0.1"
|
||||
dependencies = [
|
||||
"jni",
|
||||
"paste",
|
||||
"rand",
|
||||
"tracing",
|
||||
"tracing-android",
|
||||
"tracing-subscriber",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "log"
|
||||
version = "0.4.27"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "13dc2df351e3202783a1fe0d44375f7295ffb4049267b0f3018346dc122a1d94"
|
||||
|
||||
[[package]]
|
||||
name = "matchers"
|
||||
version = "0.1.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8263075bb86c5a1b1427b5ae862e8889656f126e9f77c484496e8b47cf5c5558"
|
||||
dependencies = [
|
||||
"regex-automata 0.1.10",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "memchr"
|
||||
version = "2.7.4"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3"
|
||||
|
||||
[[package]]
|
||||
name = "nu-ansi-term"
|
||||
version = "0.46.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "77a8165726e8236064dbb45459242600304b42a5ea24ee2948e18e023bf7ba84"
|
||||
dependencies = [
|
||||
"overload",
|
||||
"winapi",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "once_cell"
|
||||
version = "1.21.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "42f5e15c9953c5e4ccceeb2e7382a716482c34515315f7b03532b8b4e8393d2d"
|
||||
|
||||
[[package]]
|
||||
name = "overload"
|
||||
version = "0.1.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b15813163c1d831bf4a13c3610c05c0d03b39feb07f7e09fa234dac9b15aaf39"
|
||||
|
||||
[[package]]
|
||||
name = "paste"
|
||||
version = "1.0.15"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a"
|
||||
|
||||
[[package]]
|
||||
name = "pin-project-lite"
|
||||
version = "0.2.16"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3b3cff922bd51709b605d9ead9aa71031d81447142d828eb4a6eba76fe619f9b"
|
||||
|
||||
[[package]]
|
||||
name = "ppv-lite86"
|
||||
version = "0.2.21"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "85eae3c4ed2f50dcfe72643da4befc30deadb458a9b590d720cde2f2b1e97da9"
|
||||
dependencies = [
|
||||
"zerocopy",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "proc-macro2"
|
||||
version = "1.0.94"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a31971752e70b8b2686d7e46ec17fb38dad4051d94024c88df49b667caea9c84"
|
||||
dependencies = [
|
||||
"unicode-ident",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "quote"
|
||||
version = "1.0.40"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "1885c039570dc00dcb4ff087a89e185fd56bae234ddc7f056a945bf36467248d"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "r-efi"
|
||||
version = "5.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "74765f6d916ee2faa39bc8e68e4f3ed8949b48cccdac59983d287a7cb71ce9c5"
|
||||
|
||||
[[package]]
|
||||
name = "rand"
|
||||
version = "0.9.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "3779b94aeb87e8bd4e834cee3650289ee9e0d5677f976ecdb6d219e5f4f6cd94"
|
||||
dependencies = [
|
||||
"rand_chacha",
|
||||
"rand_core",
|
||||
"zerocopy",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rand_chacha"
|
||||
version = "0.9.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d3022b5f1df60f26e1ffddd6c66e8aa15de382ae63b3a0c1bfc0e4d3e3f325cb"
|
||||
dependencies = [
|
||||
"ppv-lite86",
|
||||
"rand_core",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "rand_core"
|
||||
version = "0.9.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "99d9a13982dcf210057a8a78572b2217b667c3beacbf3a0d8b454f6f82837d38"
|
||||
dependencies = [
|
||||
"getrandom",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "regex"
|
||||
version = "1.11.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b544ef1b4eac5dc2db33ea63606ae9ffcfac26c1416a2806ae0bf5f56b201191"
|
||||
dependencies = [
|
||||
"aho-corasick",
|
||||
"memchr",
|
||||
"regex-automata 0.4.9",
|
||||
"regex-syntax 0.8.5",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "regex-automata"
|
||||
version = "0.1.10"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "6c230d73fb8d8c1b9c0b3135c5142a8acee3a0558fb8db5cf1cb65f8d7862132"
|
||||
dependencies = [
|
||||
"regex-syntax 0.6.29",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "regex-automata"
|
||||
version = "0.4.9"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "809e8dc61f6de73b46c85f4c96486310fe304c434cfa43669d7b40f711150908"
|
||||
dependencies = [
|
||||
"aho-corasick",
|
||||
"memchr",
|
||||
"regex-syntax 0.8.5",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "regex-syntax"
|
||||
version = "0.6.29"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "f162c6dd7b008981e4d40210aca20b4bd0f9b60ca9271061b07f78537722f2e1"
|
||||
|
||||
[[package]]
|
||||
name = "regex-syntax"
|
||||
version = "0.8.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c"
|
||||
|
||||
[[package]]
|
||||
name = "same-file"
|
||||
version = "1.0.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "93fc1dc3aaa9bfed95e02e6eadabb4baf7e3078b0bd1b4d7b6b0b68378900502"
|
||||
dependencies = [
|
||||
"winapi-util",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "sharded-slab"
|
||||
version = "0.1.7"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "f40ca3c46823713e0d4209592e8d6e826aa57e928f09752619fc696c499637f6"
|
||||
dependencies = [
|
||||
"lazy_static",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "smallvec"
|
||||
version = "1.14.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "7fcf8323ef1faaee30a44a340193b1ac6814fd9b7b4e88e9d4519a3e4abe1cfd"
|
||||
|
||||
[[package]]
|
||||
name = "syn"
|
||||
version = "2.0.100"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b09a44accad81e1ba1cd74a32461ba89dee89095ba17b32f5d03683b1b1fc2a0"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"unicode-ident",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "thiserror"
|
||||
version = "1.0.69"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b6aaf5339b578ea85b50e080feb250a3e8ae8cfcdff9a461c9ec2904bc923f52"
|
||||
dependencies = [
|
||||
"thiserror-impl",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "thiserror-impl"
|
||||
version = "1.0.69"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "4fee6c4efc90059e10f81e6d42c60a18f76588c3d74cb83a0b242a2b6c7504c1"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "thread_local"
|
||||
version = "1.1.8"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8b9ef9bad013ada3808854ceac7b46812a6465ba368859a37e2100283d2d719c"
|
||||
dependencies = [
|
||||
"cfg-if",
|
||||
"once_cell",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tracing"
|
||||
version = "0.1.41"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "784e0ac535deb450455cbfa28a6f0df145ea1bb7ae51b821cf5e7927fdcfbdd0"
|
||||
dependencies = [
|
||||
"pin-project-lite",
|
||||
"tracing-attributes",
|
||||
"tracing-core",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tracing-android"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "12612be8f868a09c0ceae7113ff26afe79d81a24473a393cb9120ece162e86c0"
|
||||
dependencies = [
|
||||
"android_log-sys",
|
||||
"tracing",
|
||||
"tracing-subscriber",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tracing-attributes"
|
||||
version = "0.1.28"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "395ae124c09f9e6918a2310af6038fba074bcf474ac352496d5910dd59a2226d"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tracing-core"
|
||||
version = "0.1.33"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e672c95779cf947c5311f83787af4fa8fffd12fb27e4993211a84bdfd9610f9c"
|
||||
dependencies = [
|
||||
"once_cell",
|
||||
"valuable",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tracing-log"
|
||||
version = "0.2.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ee855f1f400bd0e5c02d150ae5de3840039a3f54b025156404e34c23c03f47c3"
|
||||
dependencies = [
|
||||
"log",
|
||||
"once_cell",
|
||||
"tracing-core",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "tracing-subscriber"
|
||||
version = "0.3.19"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e8189decb5ac0fa7bc8b96b7cb9b2701d60d48805aca84a238004d665fcc4008"
|
||||
dependencies = [
|
||||
"matchers",
|
||||
"nu-ansi-term",
|
||||
"once_cell",
|
||||
"regex",
|
||||
"sharded-slab",
|
||||
"smallvec",
|
||||
"thread_local",
|
||||
"tracing",
|
||||
"tracing-core",
|
||||
"tracing-log",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "unicode-ident"
|
||||
version = "1.0.18"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5a5f39404a5da50712a4c1eecf25e90dd62b613502b7e925fd4e4d19b5c96512"
|
||||
|
||||
[[package]]
|
||||
name = "valuable"
|
||||
version = "0.1.1"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ba73ea9cf16a25df0c8caa16c51acb937d5712a8429db78a3ee29d5dcacd3a65"
|
||||
|
||||
[[package]]
|
||||
name = "walkdir"
|
||||
version = "2.5.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "29790946404f91d9c5d06f9874efddea1dc06c5efe94541a7d6863108e3a5e4b"
|
||||
dependencies = [
|
||||
"same-file",
|
||||
"winapi-util",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "wasi"
|
||||
version = "0.14.2+wasi-0.2.4"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "9683f9a5a998d873c0d21fcbe3c083009670149a8fab228644b8bd36b2c48cb3"
|
||||
dependencies = [
|
||||
"wit-bindgen-rt",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "winapi"
|
||||
version = "0.3.9"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419"
|
||||
dependencies = [
|
||||
"winapi-i686-pc-windows-gnu",
|
||||
"winapi-x86_64-pc-windows-gnu",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "winapi-i686-pc-windows-gnu"
|
||||
version = "0.4.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
|
||||
|
||||
[[package]]
|
||||
name = "winapi-util"
|
||||
version = "0.1.9"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb"
|
||||
dependencies = [
|
||||
"windows-sys 0.59.0",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "winapi-x86_64-pc-windows-gnu"
|
||||
version = "0.4.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
|
||||
|
||||
[[package]]
|
||||
name = "windows-sys"
|
||||
version = "0.45.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "75283be5efb2831d37ea142365f009c02ec203cd29a3ebecbc093d52315b66d0"
|
||||
dependencies = [
|
||||
"windows-targets 0.42.2",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows-sys"
|
||||
version = "0.59.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "1e38bc4d79ed67fd075bcc251a1c39b32a1776bbe92e5bef1f0bf1f8c531853b"
|
||||
dependencies = [
|
||||
"windows-targets 0.52.6",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows-targets"
|
||||
version = "0.42.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8e5180c00cd44c9b1c88adb3693291f1cd93605ded80c250a75d472756b4d071"
|
||||
dependencies = [
|
||||
"windows_aarch64_gnullvm 0.42.2",
|
||||
"windows_aarch64_msvc 0.42.2",
|
||||
"windows_i686_gnu 0.42.2",
|
||||
"windows_i686_msvc 0.42.2",
|
||||
"windows_x86_64_gnu 0.42.2",
|
||||
"windows_x86_64_gnullvm 0.42.2",
|
||||
"windows_x86_64_msvc 0.42.2",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows-targets"
|
||||
version = "0.52.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "9b724f72796e036ab90c1021d4780d4d3d648aca59e491e6b98e725b84e99973"
|
||||
dependencies = [
|
||||
"windows_aarch64_gnullvm 0.52.6",
|
||||
"windows_aarch64_msvc 0.52.6",
|
||||
"windows_i686_gnu 0.52.6",
|
||||
"windows_i686_gnullvm",
|
||||
"windows_i686_msvc 0.52.6",
|
||||
"windows_x86_64_gnu 0.52.6",
|
||||
"windows_x86_64_gnullvm 0.52.6",
|
||||
"windows_x86_64_msvc 0.52.6",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows_aarch64_gnullvm"
|
||||
version = "0.42.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "597a5118570b68bc08d8d59125332c54f1ba9d9adeedeef5b99b02ba2b0698f8"
|
||||
|
||||
[[package]]
|
||||
name = "windows_aarch64_gnullvm"
|
||||
version = "0.52.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3"
|
||||
|
||||
[[package]]
|
||||
name = "windows_aarch64_msvc"
|
||||
version = "0.42.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e08e8864a60f06ef0d0ff4ba04124db8b0fb3be5776a5cd47641e942e58c4d43"
|
||||
|
||||
[[package]]
|
||||
name = "windows_aarch64_msvc"
|
||||
version = "0.52.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469"
|
||||
|
||||
[[package]]
|
||||
name = "windows_i686_gnu"
|
||||
version = "0.42.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c61d927d8da41da96a81f029489353e68739737d3beca43145c8afec9a31a84f"
|
||||
|
||||
[[package]]
|
||||
name = "windows_i686_gnu"
|
||||
version = "0.52.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8e9b5ad5ab802e97eb8e295ac6720e509ee4c243f69d781394014ebfe8bbfa0b"
|
||||
|
||||
[[package]]
|
||||
name = "windows_i686_gnullvm"
|
||||
version = "0.52.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66"
|
||||
|
||||
[[package]]
|
||||
name = "windows_i686_msvc"
|
||||
version = "0.42.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "44d840b6ec649f480a41c8d80f9c65108b92d89345dd94027bfe06ac444d1060"
|
||||
|
||||
[[package]]
|
||||
name = "windows_i686_msvc"
|
||||
version = "0.52.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_gnu"
|
||||
version = "0.42.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "8de912b8b8feb55c064867cf047dda097f92d51efad5b491dfb98f6bbb70cb36"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_gnu"
|
||||
version = "0.52.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "147a5c80aabfbf0c7d901cb5895d1de30ef2907eb21fbbab29ca94c5b08b1a78"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_gnullvm"
|
||||
version = "0.42.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "26d41b46a36d453748aedef1486d5c7a85db22e56aff34643984ea85514e94a3"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_gnullvm"
|
||||
version = "0.52.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_msvc"
|
||||
version = "0.42.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "9aec5da331524158c6d1a4ac0ab1541149c0b9505fde06423b02f5ef0106b9f0"
|
||||
|
||||
[[package]]
|
||||
name = "windows_x86_64_msvc"
|
||||
version = "0.52.6"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec"
|
||||
|
||||
[[package]]
|
||||
name = "wit-bindgen-rt"
|
||||
version = "0.39.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "6f42320e61fe2cfd34354ecb597f86f413484a798ba44a8ca1165c58d42da6c1"
|
||||
dependencies = [
|
||||
"bitflags",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "zerocopy"
|
||||
version = "0.8.24"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "2586fea28e186957ef732a5f8b3be2da217d65c5969d4b1e17f973ebbe876879"
|
||||
dependencies = [
|
||||
"zerocopy-derive",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "zerocopy-derive"
|
||||
version = "0.8.24"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "a996a8f63c5c4448cd959ac1bab0aaa3306ccfd060472f85943ee0750f0169be"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
"syn",
|
||||
]
|
16
Cargo.toml
Normal file
16
Cargo.toml
Normal file
|
@ -0,0 +1,16 @@
|
|||
[workspace]
|
||||
members = ["libserver"]
|
||||
resolver = "2"
|
||||
|
||||
[workspace.package]
|
||||
version = "0.0.1"
|
||||
edition = "2024"
|
||||
|
||||
[workspace.dependencies]
|
||||
jni = "0.21.1"
|
||||
tracing = "0.1.41"
|
||||
tracing-android = "0.2.0"
|
||||
tracing-subscriber = { version = "0.3.19", features = ["env-filter"] }
|
||||
|
||||
rand = "0.9"
|
||||
paste = "1.0.15"
|
50
README.md
50
README.md
|
@ -1,3 +1,49 @@
|
|||
# Scroll
|
||||
# scroll-rs
|
||||
Experimental Clash Royale server emulator on top of libg.so (v1.3.2)
|
||||
# 
|
||||
|
||||
Experimental Clash Royale server on top of libg.so
|
||||
# Why?
|
||||
Supercell's games have their game logic implementation included in both client and server, for independent execution (the `Logic*` family of classes is shared). However, to implement a feature-complete server emulator, you should rewrite the logic entirely. This experimental way is to use logic that is retained in the game, basically turning client into a server.
|
||||
|
||||
# Current features
|
||||
- All cards are unlocked
|
||||
- Training battles
|
||||
- More features are planned to be implemented (including PvP battles), stay tuned
|
||||
|
||||
# Implementation
|
||||
Server side code is written in Rust. We provide idiomatic bindings for the structures/functions from `libg.so`.
|
||||
|
||||
# Getting started
|
||||
#### NOTE: you have to use a device with support of armeabi-v7a binaries
|
||||
### a) Using pre-built apk files
|
||||
Navigate to the [Releases](https://git.xeondev.com/Supercell/Scroll/releases) page and download both Server and Client APK files (by default, they're targeted to the `127.0.0.1:9339` endpoint). Next, install both of them. Open the server application first (it should stay with black screen), then leave it running in the background. Open the client and play!
|
||||
### b) Building from sources
|
||||
#### Requirements:
|
||||
- [Rust 1.85+](https://www.rust-lang.org/tools/install)
|
||||
- [Android NDK](https://developer.android.com/ndk/downloads)
|
||||
- [cargo-ndk](https://docs.rs/crate/cargo-ndk/3.5.4)
|
||||
|
||||
#### Preparing the toolchain
|
||||
- Install android armv7 target via rustup:
|
||||
```sh
|
||||
rustup target add armv7-linux-androideabi
|
||||
```
|
||||
- Install cargo-ndk extension:
|
||||
```sh
|
||||
cargo install cargo-ndk
|
||||
```
|
||||
|
||||
##### NOTE: make sure you have configured the `ANDROID_NDK_HOME` environment variable before invoking build command.
|
||||
|
||||
#### Compiling server library
|
||||
```sh
|
||||
git clone https://git.xeondev.com/Supercell/Scroll
|
||||
cd Scroll
|
||||
cargo ndk -t armeabi-v7a build --release
|
||||
```
|
||||
|
||||
### Community
|
||||
[Our Discord Server](https://discord.gg/reversedrooms) is open for everyone who's interested in our projects!
|
||||
|
||||
### Support
|
||||
Your support for this project is greatly appreciated! If you'd like to contribute, feel free to send a tip [via Boosty](https://boosty.to/xeondev/donate)!
|
||||
|
|
18
libserver/Cargo.toml
Normal file
18
libserver/Cargo.toml
Normal file
|
@ -0,0 +1,18 @@
|
|||
[package]
|
||||
name = "libserver"
|
||||
version.workspace = true
|
||||
edition.workspace = true
|
||||
|
||||
[lib]
|
||||
name = "server"
|
||||
crate-type = ["cdylib"]
|
||||
|
||||
[dependencies]
|
||||
jni.workspace = true
|
||||
|
||||
tracing.workspace = true
|
||||
tracing-android.workspace = true
|
||||
tracing-subscriber.workspace = true
|
||||
|
||||
rand.workspace = true
|
||||
paste.workspace = true
|
18
libserver/src/allocator.rs
Normal file
18
libserver/src/allocator.rs
Normal file
|
@ -0,0 +1,18 @@
|
|||
mod libc_extern {
|
||||
unsafe extern "C" {
|
||||
pub fn malloc(amount: usize) -> *const u8;
|
||||
pub fn free(ptr: *const u8);
|
||||
}
|
||||
}
|
||||
|
||||
pub fn malloc(amount: usize) -> *const u8 {
|
||||
unsafe { libc_extern::malloc(amount) }
|
||||
}
|
||||
|
||||
pub fn free<T>(ptr: *const T) {
|
||||
unsafe { libc_extern::free(ptr as *const u8) }
|
||||
}
|
||||
|
||||
pub fn new<T>() -> *mut T {
|
||||
malloc(std::mem::size_of::<T>()).cast::<T>().cast_mut()
|
||||
}
|
15
libserver/src/array_list.rs
Normal file
15
libserver/src/array_list.rs
Normal file
|
@ -0,0 +1,15 @@
|
|||
#[repr(C)]
|
||||
pub struct LogicArrayList<T> {
|
||||
pub data: *const T,
|
||||
pub _capacity: usize,
|
||||
pub count: usize,
|
||||
}
|
||||
|
||||
impl<T> LogicArrayList<T>
|
||||
where
|
||||
T: Sized,
|
||||
{
|
||||
pub fn as_slice(&self) -> &[T] {
|
||||
unsafe { std::slice::from_raw_parts(self.data, self.count) }
|
||||
}
|
||||
}
|
57
libserver/src/ffi_util.rs
Normal file
57
libserver/src/ffi_util.rs
Normal file
|
@ -0,0 +1,57 @@
|
|||
use std::sync::LazyLock;
|
||||
|
||||
pub static LIBG_BASE: LazyLock<usize> =
|
||||
LazyLock::new(|| get_module_base("libg.so").expect("failed to get libg.so base address"));
|
||||
|
||||
#[repr(transparent)]
|
||||
pub struct Nullable<T>(*const T);
|
||||
|
||||
impl<T> Nullable<T> {
|
||||
pub fn get(&self) -> Option<&T> {
|
||||
(!self.0.is_null()).then(|| unsafe { &*self.0 })
|
||||
}
|
||||
|
||||
pub fn get_mut(&mut self) -> Option<&mut T> {
|
||||
(!self.0.is_null()).then(|| unsafe { &mut *self.0.cast_mut() })
|
||||
}
|
||||
|
||||
pub fn set(&mut self, value: *const T) {
|
||||
self.0 = value;
|
||||
}
|
||||
}
|
||||
|
||||
macro_rules! import {
|
||||
($name:ident($($arg_name:ident: $arg_type:ty),*) -> $ret_type:ty = $rva:expr) => {
|
||||
pub fn $name($($arg_name: $arg_type,)*) -> $ret_type {
|
||||
unsafe {
|
||||
type FuncType = unsafe extern "cdecl" fn($($arg_type,)*) -> $ret_type;
|
||||
::std::mem::transmute::<usize, FuncType>(*crate::ffi_util::LIBG_BASE + $rva)($($arg_name,)*)
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
pub(crate) use import;
|
||||
|
||||
pub fn get_module_base(shared_object_name: &str) -> Option<usize> {
|
||||
use std::{
|
||||
fs::File,
|
||||
io::{self, BufRead},
|
||||
};
|
||||
|
||||
let file = File::open("/proc/self/maps").expect("failed to open /proc/self/maps");
|
||||
let reader = io::BufReader::new(file);
|
||||
|
||||
for line in reader.lines() {
|
||||
let line = line.ok()?;
|
||||
if line.contains(shared_object_name) {
|
||||
let address_str = line.split_whitespace().next().unwrap_or("");
|
||||
let address =
|
||||
usize::from_str_radix(&address_str.split('-').next().unwrap_or(""), 16).ok()?;
|
||||
|
||||
return Some(address);
|
||||
}
|
||||
}
|
||||
|
||||
None
|
||||
}
|
82
libserver/src/lib.rs
Normal file
82
libserver/src/lib.rs
Normal file
|
@ -0,0 +1,82 @@
|
|||
use std::{
|
||||
ffi::c_void,
|
||||
net::{SocketAddr, TcpListener},
|
||||
os::fd::IntoRawFd,
|
||||
sync::LazyLock,
|
||||
thread,
|
||||
};
|
||||
|
||||
use network::{LogicScrollMessageFactory, Messaging, RC4Encrypter};
|
||||
use session::PlayerSession;
|
||||
use tracing::info;
|
||||
|
||||
mod allocator;
|
||||
mod array_list;
|
||||
mod ffi_util;
|
||||
mod logic;
|
||||
mod network;
|
||||
mod resources;
|
||||
mod sc_string;
|
||||
mod session;
|
||||
mod util;
|
||||
|
||||
#[allow(non_snake_case)]
|
||||
#[unsafe(no_mangle)]
|
||||
pub unsafe extern "system" fn JNI_OnLoad(_vm: jni::JavaVM, _: *mut c_void) -> jni::sys::jint {
|
||||
init_tracing();
|
||||
|
||||
thread::spawn(|| server_main());
|
||||
|
||||
jni::sys::JNI_VERSION_1_6
|
||||
}
|
||||
|
||||
fn server_main() {
|
||||
const TCP_ADDR: &str = "127.0.0.1:9339";
|
||||
|
||||
info!("starting server...");
|
||||
resources::init();
|
||||
info!("resources initialized");
|
||||
|
||||
let listener = TcpListener::bind(TCP_ADDR).unwrap();
|
||||
while let Ok((stream, addr)) = listener.accept() {
|
||||
info!("new connection from {addr}");
|
||||
let fd = stream.into_raw_fd();
|
||||
|
||||
thread::spawn(move || receive_loop(fd, addr));
|
||||
}
|
||||
}
|
||||
|
||||
fn receive_loop(fd: i32, addr: SocketAddr) {
|
||||
static MESSAGE_FACTORY: LazyLock<LogicScrollMessageFactory> =
|
||||
LazyLock::new(|| LogicScrollMessageFactory::new(false));
|
||||
|
||||
let mut messaging = Messaging::new(80);
|
||||
|
||||
let encrypter = RC4Encrypter::new(LogicScrollMessageFactory::RC4_KEY, "nonce");
|
||||
let decrypter = RC4Encrypter::new(LogicScrollMessageFactory::RC4_KEY, "nonce");
|
||||
|
||||
messaging.get_connection_mut().fd = fd;
|
||||
messaging.get_connection_mut().is_connected = true;
|
||||
messaging.set_encrypters(encrypter, decrypter);
|
||||
messaging.set_message_factory(&MESSAGE_FACTORY);
|
||||
|
||||
let mut session = PlayerSession::new(messaging);
|
||||
session.work_until_disconnect();
|
||||
|
||||
info!("client from {addr} disconnected");
|
||||
}
|
||||
|
||||
fn init_tracing() {
|
||||
use tracing::level_filters::LevelFilter;
|
||||
use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt};
|
||||
|
||||
tracing_subscriber::registry()
|
||||
.with(tracing_subscriber::fmt::layer())
|
||||
.with(tracing_android::layer("SCROLL-SERVER").unwrap())
|
||||
.with(
|
||||
tracing_subscriber::EnvFilter::builder()
|
||||
.with_default_directive(LevelFilter::INFO.into())
|
||||
.from_env_lossy(),
|
||||
)
|
||||
.init();
|
||||
}
|
63
libserver/src/logic/avatar.rs
Normal file
63
libserver/src/logic/avatar.rs
Normal file
|
@ -0,0 +1,63 @@
|
|||
use super::{data::LogicData, math::LogicLong};
|
||||
use crate::{
|
||||
allocator,
|
||||
ffi_util::import,
|
||||
sc_string::ScString,
|
||||
util::data_stream::{ByteStream, ChecksumEncoder},
|
||||
};
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicAvatarVTable {
|
||||
finalizer_1: extern "C" fn(*mut LogicClientAvatar),
|
||||
finalizer_2: extern "C" fn(*mut LogicClientAvatar),
|
||||
pub destruct: extern "C" fn(*mut LogicClientAvatar),
|
||||
pub encode: extern "C" fn(*const LogicClientAvatar, *mut ChecksumEncoder),
|
||||
pub decode: extern "C" fn(*mut LogicClientAvatar, *mut ByteStream),
|
||||
set_logic_game_mode: extern "C" fn(*mut LogicClientAvatar, usize),
|
||||
is_npc_avatar: extern "C" fn(*const LogicClientAvatar) -> bool,
|
||||
get_checksum: extern "C" fn(*const LogicClientAvatar) -> i32,
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicClientAvatar {
|
||||
pub vtable: &'static LogicAvatarVTable,
|
||||
pub logic_game_mode: usize,
|
||||
pub id: LogicLong, // 8
|
||||
pub account_id: LogicLong, // 16
|
||||
pub home_id: LogicLong, // 24
|
||||
unk_gap_1: [u8; 28], // 32 - 60
|
||||
pub name: ScString, // 60
|
||||
pub score: i32, // 80
|
||||
unk_gap_2: [u8; 28], // 84 - 112
|
||||
pub level: i32, // 112
|
||||
unk_gap_3: [u8; 68], // 116 - 184
|
||||
pub npc_win_count: i32, // 184
|
||||
pub npc_lose_count: i32, // 188,
|
||||
unk_192: bool,
|
||||
pub name_set_by_user: bool,
|
||||
unk_194: bool,
|
||||
pub name_change_state: i32, // 196
|
||||
pub arena: &'static LogicData, // 200
|
||||
}
|
||||
|
||||
impl LogicClientAvatar {
|
||||
pub fn new() -> &'static mut LogicClientAvatar {
|
||||
import!(logic_client_avatar_ctor(ptr: *mut LogicClientAvatar) -> () = 0xEAE00 + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
logic_client_avatar_ctor(instance);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
|
||||
pub fn clone(&self) -> &'static mut LogicClientAvatar {
|
||||
let mut stream = ByteStream::new(128);
|
||||
|
||||
(self.vtable.encode)(self, &mut stream.parent);
|
||||
stream.reset_offset();
|
||||
|
||||
let cloned_avatar = LogicClientAvatar::new();
|
||||
(cloned_avatar.vtable.decode)(cloned_avatar, &mut stream);
|
||||
stream.destruct();
|
||||
|
||||
cloned_avatar
|
||||
}
|
||||
}
|
28
libserver/src/logic/battle.rs
Normal file
28
libserver/src/logic/battle.rs
Normal file
|
@ -0,0 +1,28 @@
|
|||
use crate::ffi_util::import;
|
||||
|
||||
use super::{
|
||||
data::{LogicData, LogicLocationData, LogicNpcData},
|
||||
spell::LogicSpellDeck,
|
||||
};
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicBattle {
|
||||
unk_gap_1: [u8; 8], // 0 - 8,
|
||||
pub battle_type: i32, // 8
|
||||
unk_gap_2: [u8; 32], // 12 - 44
|
||||
pub npc_data: &'static LogicNpcData,
|
||||
pub arena_data: &'static LogicData,
|
||||
}
|
||||
|
||||
impl LogicBattle {
|
||||
pub fn set_location(&mut self, location: &LogicLocationData, skip_init: bool, listener: usize) {
|
||||
import!(logic_battle_set_location(battle: *mut LogicBattle, location: *const LogicLocationData, skip_init: bool, listener: usize) -> () = 0xEE05C + 1);
|
||||
|
||||
logic_battle_set_location(self, location, skip_init, listener);
|
||||
}
|
||||
|
||||
pub fn set_spell_decks(&mut self, deck_1: &LogicSpellDeck, deck_2: &LogicSpellDeck) {
|
||||
import!(logic_battle_set_spell_decks(battle: *mut LogicBattle, deck_1: *const LogicSpellDeck, deck_2: *const LogicSpellDeck) -> () = 0xEE17C + 1);
|
||||
logic_battle_set_spell_decks(self, deck_1, deck_2);
|
||||
}
|
||||
}
|
15
libserver/src/logic/command/manager.rs
Normal file
15
libserver/src/logic/command/manager.rs
Normal file
|
@ -0,0 +1,15 @@
|
|||
use crate::ffi_util::import;
|
||||
|
||||
use super::LogicCommand;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicCommandManager {
|
||||
unk_gap_1: [u8; 0x14],
|
||||
}
|
||||
|
||||
impl LogicCommandManager {
|
||||
pub fn add_command(&mut self, command: &LogicCommand) {
|
||||
import!(logic_command_manager_add_command(ptr: *mut LogicCommandManager, command: *const LogicCommand) -> () = 0xF0914 + 1);
|
||||
logic_command_manager_add_command(self, command);
|
||||
}
|
||||
}
|
33
libserver/src/logic/command/mod.rs
Normal file
33
libserver/src/logic/command/mod.rs
Normal file
|
@ -0,0 +1,33 @@
|
|||
use crate::util::data_stream::{ByteStream, ChecksumEncoder};
|
||||
|
||||
use super::{json::LogicJSONNode, math::LogicLong, mode::LogicGameMode};
|
||||
|
||||
mod manager;
|
||||
pub use manager::LogicCommandManager;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicCommandVTable {
|
||||
pub destruct: extern "C" fn(ptr: *mut LogicCommand),
|
||||
finalizer_1: extern "C" fn(ptr: *mut LogicCommand),
|
||||
finalizer_2: extern "C" fn(ptr: *mut LogicCommand),
|
||||
pub execute: extern "C" fn(ptr: *const LogicCommand, logic_game_mode: *mut LogicGameMode),
|
||||
pub get_command_type: extern "C" fn() -> i16,
|
||||
pub encode: extern "C" fn(ptr: *const LogicCommand, encoder: *mut ChecksumEncoder),
|
||||
pub decode: extern "C" fn(ptr: *mut LogicCommand, stream: *mut ByteStream),
|
||||
pub get_command_gold_cost:
|
||||
extern "C" fn(ptr: *const LogicCommand, logic_game_mode: *const LogicGameMode) -> i32,
|
||||
pub get_command_diamond_cost:
|
||||
extern "C" fn(ptr: *const LogicCommand, logic_game_mode: *const LogicGameMode) -> i32,
|
||||
pub get_json_for_replay: extern "C" fn(ptr: *const LogicCommand) -> *const LogicJSONNode,
|
||||
pub load_from_json: extern "C" fn(ptr: *mut LogicCommand, json: *const LogicJSONNode),
|
||||
pub is_saved_for_replay: extern "C" fn(ptr: *const LogicCommand) -> bool,
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicCommand {
|
||||
pub vtable: &'static LogicCommandVTable,
|
||||
unk: bool,
|
||||
pub execute_tick: i32,
|
||||
pub tick_when_given: i32,
|
||||
pub executor_account_id: LogicLong,
|
||||
}
|
64
libserver/src/logic/data.rs
Normal file
64
libserver/src/logic/data.rs
Normal file
|
@ -0,0 +1,64 @@
|
|||
use crate::{ffi_util::import, resources::CSVNode, sc_string::ScString};
|
||||
|
||||
pub struct LogicDataTables;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicData {
|
||||
unk_gap_1: [u8; 16],
|
||||
pub global_id: i32, // 16
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicNpcData {
|
||||
pub parent: LogicData,
|
||||
unk_gap_1: [u8; 12], // 20 - 32
|
||||
pub location: &'static LogicLocationData,
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicLocationData {
|
||||
pub parent: LogicData,
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicSpellData {
|
||||
pub parent: LogicData,
|
||||
}
|
||||
|
||||
impl LogicLocationData {
|
||||
pub fn get_tilemap_file_name(&self) -> String {
|
||||
import!(logic_location_data_get_tilemap_file_name(data: *const LogicLocationData) -> *const ScString = 0x103EC4 + 1);
|
||||
let file_name = logic_location_data_get_tilemap_file_name(self);
|
||||
unsafe { (&*file_name).to_string() }
|
||||
}
|
||||
}
|
||||
|
||||
impl LogicSpellData {
|
||||
pub fn get_max_level_index(&self) -> i32 {
|
||||
import!(logic_spell_data_get_max_level_index(data: *const LogicSpellData) -> i32 = 0x10A2F4 + 1);
|
||||
logic_spell_data_get_max_level_index(self)
|
||||
}
|
||||
}
|
||||
|
||||
impl LogicDataTables {
|
||||
pub fn get_data_by_id(global_id: i32) -> Option<&'static LogicData> {
|
||||
import!(logic_data_tables_get_data_by_id(global_id: i32) -> *const LogicData = 0xFCB88 + 1);
|
||||
let data = logic_data_tables_get_data_by_id(global_id);
|
||||
(!data.is_null()).then(|| unsafe { &*data })
|
||||
}
|
||||
|
||||
pub fn get_spell_count() -> usize {
|
||||
import!(logic_data_tables_get_spell_count() -> usize = 0xFE974 + 1);
|
||||
logic_data_tables_get_spell_count()
|
||||
}
|
||||
|
||||
pub fn get_spell_at(index: usize) -> &'static LogicSpellData {
|
||||
import!(logic_data_tables_get_spell_at(index: usize) -> *const LogicSpellData = 0xFE98C + 1);
|
||||
unsafe { &*logic_data_tables_get_spell_at(index) }
|
||||
}
|
||||
|
||||
pub fn init_tilemap(csv: CSVNode, id: i32) {
|
||||
import!(logic_data_tables_init_tilemap(csv: *const u8, id: i32) -> () = 0xFE49C + 1);
|
||||
logic_data_tables_init_tilemap(csv.0, id);
|
||||
}
|
||||
}
|
68
libserver/src/logic/home.rs
Normal file
68
libserver/src/logic/home.rs
Normal file
|
@ -0,0 +1,68 @@
|
|||
use crate::{
|
||||
allocator,
|
||||
ffi_util::{import, Nullable},
|
||||
util::data_stream::{ByteStream, ChecksumEncoder},
|
||||
};
|
||||
|
||||
use super::{
|
||||
data::LogicSpellData,
|
||||
math::LogicLong,
|
||||
spell::{LogicSpellCollection, LogicSpellDeck},
|
||||
};
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicClientHome {
|
||||
pub change_listener: &'static LogicHomeChangeListener,
|
||||
pub logic_game_mode: usize,
|
||||
pub id: Nullable<LogicLong>, // 8
|
||||
pub spell_deck: LogicSpellDeck, // 12
|
||||
pub spell_collection: LogicSpellCollection, // 44
|
||||
unk_gap_1: [u8; 332], // 60 - 392
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicHomeChangeListener {
|
||||
pub _vtable: usize,
|
||||
}
|
||||
|
||||
impl LogicClientHome {
|
||||
pub fn new() -> &'static mut Self {
|
||||
import!(logic_client_home_ctor(ptr: *mut LogicClientHome) -> () = 0x114D40 + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
logic_client_home_ctor(instance);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
|
||||
pub fn has_spell(&self, spell_data: &LogicSpellData) -> bool {
|
||||
import!(logic_client_home_has_spell(ptr: *const LogicClientHome, spell_data: *const LogicSpellData) -> bool = 0x116EDC + 1);
|
||||
logic_client_home_has_spell(self, spell_data)
|
||||
}
|
||||
|
||||
pub fn encode(&self, encoder: &mut ChecksumEncoder) {
|
||||
import!(logic_client_home_encode(ptr: *const LogicClientHome, encoder: *mut ChecksumEncoder) -> () = 0x115CD4 + 1);
|
||||
logic_client_home_encode(self, encoder);
|
||||
}
|
||||
|
||||
pub fn decode(&mut self, stream: &mut ByteStream) {
|
||||
import!(logic_client_home_decode(ptr: *mut LogicClientHome, stream: *mut ByteStream) -> () = 0x115FD0 + 1);
|
||||
logic_client_home_decode(self, stream);
|
||||
}
|
||||
|
||||
pub fn destruct(&self) {
|
||||
import!(logic_client_home_destruct(ptr: *const LogicClientHome) -> () = 0x115058 + 1);
|
||||
logic_client_home_destruct(self);
|
||||
}
|
||||
|
||||
pub fn clone(&self) -> &'static mut LogicClientHome {
|
||||
let mut stream = ByteStream::new(128);
|
||||
|
||||
self.encode(&mut stream.parent);
|
||||
stream.reset_offset();
|
||||
|
||||
let cloned_home = LogicClientHome::new();
|
||||
cloned_home.decode(&mut stream);
|
||||
stream.destruct();
|
||||
|
||||
cloned_home
|
||||
}
|
||||
}
|
16
libserver/src/logic/json.rs
Normal file
16
libserver/src/logic/json.rs
Normal file
|
@ -0,0 +1,16 @@
|
|||
use crate::{ffi_util::import, sc_string::ScString};
|
||||
|
||||
pub struct LogicJSONParser;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicJSONNode {
|
||||
vtable: usize,
|
||||
}
|
||||
|
||||
impl LogicJSONParser {
|
||||
pub fn parse(data: &str) -> Option<&'static LogicJSONNode> {
|
||||
import!(logic_json_parser_parse(data: *const ScString) -> *const LogicJSONNode = 0x17D2E4 + 1);
|
||||
let result = logic_json_parser_parse(ScString::new(data));
|
||||
(!result.is_null()).then(|| unsafe { &*result })
|
||||
}
|
||||
}
|
31
libserver/src/logic/math.rs
Normal file
31
libserver/src/logic/math.rs
Normal file
|
@ -0,0 +1,31 @@
|
|||
use std::fmt;
|
||||
|
||||
use crate::allocator;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicLong {
|
||||
pub higher_int: i32,
|
||||
pub lower_int: i32,
|
||||
}
|
||||
|
||||
impl LogicLong {
|
||||
pub fn new(higher_int: i32, lower_int: i32) -> &'static Self {
|
||||
unsafe {
|
||||
let instance = &mut *allocator::new::<Self>();
|
||||
instance.higher_int = higher_int;
|
||||
instance.lower_int = lower_int;
|
||||
instance
|
||||
}
|
||||
}
|
||||
|
||||
pub fn set(&mut self, higher_int: i32, lower_int: i32) {
|
||||
self.higher_int = higher_int;
|
||||
self.lower_int = lower_int;
|
||||
}
|
||||
}
|
||||
|
||||
impl fmt::Display for LogicLong {
|
||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||
write!(f, "LogicLong({},{})", self.higher_int, self.lower_int)
|
||||
}
|
||||
}
|
17
libserver/src/logic/mod.rs
Normal file
17
libserver/src/logic/mod.rs
Normal file
|
@ -0,0 +1,17 @@
|
|||
pub mod avatar;
|
||||
pub mod battle;
|
||||
pub mod command;
|
||||
pub mod data;
|
||||
pub mod home;
|
||||
pub mod json;
|
||||
pub mod math;
|
||||
pub mod mode;
|
||||
pub mod spell;
|
||||
pub mod time;
|
||||
|
||||
pub struct LogicVersion;
|
||||
|
||||
impl LogicVersion {
|
||||
pub const MAJOR: i32 = 2;
|
||||
pub const BUILD: i32 = 1666;
|
||||
}
|
84
libserver/src/logic/mode.rs
Normal file
84
libserver/src/logic/mode.rs
Normal file
|
@ -0,0 +1,84 @@
|
|||
use crate::{
|
||||
allocator,
|
||||
ffi_util::{import, Nullable},
|
||||
util::data_stream::ChecksumEncoder,
|
||||
};
|
||||
|
||||
use super::{
|
||||
avatar::LogicClientAvatar, battle::LogicBattle, command::LogicCommandManager,
|
||||
home::LogicClientHome, time::LogicTime,
|
||||
};
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicGameMode {
|
||||
unk_gap_1: [u8; 36], // 0-36
|
||||
pub logic_command_manager: &'static mut LogicCommandManager, // 36
|
||||
game_object_manager: usize, // 40 // TODO!
|
||||
achievement_manager: usize, // 44 // TODO!
|
||||
pub home: Nullable<LogicClientHome>, // 48
|
||||
pub player_avatar: Nullable<LogicClientAvatar>, // 52
|
||||
pub time: LogicTime, // 56
|
||||
unk_gap_2: [u8; 8], // 116 - 124
|
||||
pub battle: Nullable<LogicBattle>, // 124
|
||||
unk_gap_3: [u8; 168], // 128 - 296
|
||||
}
|
||||
|
||||
impl LogicGameMode {
|
||||
pub fn new(is_battle: bool) -> &'static mut Self {
|
||||
import!(logic_game_mode_ctor(ptr: *mut LogicGameMode, is_battle: bool) -> () = 0x119DAC + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
logic_game_mode_ctor(instance, is_battle);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
|
||||
pub fn load_home_state(
|
||||
&mut self,
|
||||
logic_client_home: *const LogicClientHome,
|
||||
logic_client_avatar: *const LogicClientAvatar,
|
||||
home_passed_time_seconds: i32,
|
||||
home_passed_time_ticks: i32,
|
||||
home_random_seed: i32,
|
||||
) {
|
||||
import!(logic_game_mode_load_home_state(lgm: *mut LogicGameMode, lch: *const LogicClientHome, lca: *const LogicClientAvatar, hpts: i32, hptt: i32, hrs: i32) -> () = 0x11A274 + 1);
|
||||
|
||||
logic_game_mode_load_home_state(
|
||||
self,
|
||||
logic_client_home,
|
||||
logic_client_avatar,
|
||||
home_passed_time_seconds,
|
||||
home_passed_time_ticks,
|
||||
home_random_seed,
|
||||
);
|
||||
}
|
||||
|
||||
pub fn add_player(&mut self, player_avatar: &LogicClientAvatar) {
|
||||
import!(logic_game_mode_add_player(lgm: *mut LogicGameMode, player_avatar: *const LogicClientAvatar) -> () = 0x11AD80 + 1);
|
||||
logic_game_mode_add_player(self, player_avatar);
|
||||
}
|
||||
|
||||
pub fn update_one_tick(&mut self) {
|
||||
import!(logic_game_mode_update_one_tick(lgm: *mut LogicGameMode) -> () = 0x11A718 + 1);
|
||||
logic_game_mode_update_one_tick(self);
|
||||
}
|
||||
|
||||
pub fn calculate_checksum(&self) -> i32 {
|
||||
import!(logic_game_mode_calculate_checksum(lgm: *const LogicGameMode) -> i32 = 0x11A85C + 1);
|
||||
logic_game_mode_calculate_checksum(self)
|
||||
}
|
||||
|
||||
pub fn encode(
|
||||
&self,
|
||||
encoder: &mut ChecksumEncoder,
|
||||
include_commands: bool,
|
||||
avatar: &LogicClientAvatar,
|
||||
unk: bool,
|
||||
) {
|
||||
import!(logic_game_mode_encode(lgm: *const LogicGameMode, encoder: *mut ChecksumEncoder, include_commands: bool, avatar: *const LogicClientAvatar, unk: bool) -> () = 0x11A8D4 + 1);
|
||||
logic_game_mode_encode(self, encoder, include_commands, avatar, unk);
|
||||
}
|
||||
|
||||
pub fn destruct(&self) {
|
||||
import!(logic_game_mode_destruct(lgm: *const LogicGameMode) -> () = 0x11A034 + 1);
|
||||
logic_game_mode_destruct(self);
|
||||
}
|
||||
}
|
69
libserver/src/logic/spell.rs
Normal file
69
libserver/src/logic/spell.rs
Normal file
|
@ -0,0 +1,69 @@
|
|||
use crate::{allocator, array_list::LogicArrayList, ffi_util::import};
|
||||
|
||||
use super::{data::LogicSpellData, json::LogicJSONNode};
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicSpellDeck {
|
||||
vtable: usize,
|
||||
unk_gap_1: [u8; 28],
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicSpellCollection {
|
||||
pub spells: LogicArrayList<&'static mut LogicSpell>,
|
||||
pub current_sort: i32,
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicSpell {
|
||||
pub data: &'static LogicSpellData,
|
||||
pub level_index: i32,
|
||||
pub create_time: i32,
|
||||
pub count: i32,
|
||||
pub listener: &'static LogicSpellListener,
|
||||
pub new_upgrade: bool,
|
||||
pub new_flag: bool,
|
||||
pub new_count: i32,
|
||||
pub use_count: i32,
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicSpellListener {
|
||||
vtable: usize,
|
||||
}
|
||||
|
||||
impl LogicSpell {
|
||||
pub fn new() -> &'static mut Self {
|
||||
import!(logic_spell_ctor(ptr: *mut LogicSpell) -> () = 0x117FCC + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
logic_spell_ctor(instance);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
}
|
||||
|
||||
impl LogicSpellDeck {
|
||||
#[expect(unused)]
|
||||
pub fn new() -> &'static mut Self {
|
||||
import!(logic_spell_deck_ctor(ptr: *mut LogicSpellDeck) -> () = 0x1188FC + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
logic_spell_deck_ctor(instance);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
|
||||
pub fn load(&mut self, data: &LogicJSONNode) {
|
||||
import!(logic_spell_deck_load(ptr: *mut LogicSpellDeck, data: *const LogicJSONNode) -> () = 0x118A7C + 1);
|
||||
logic_spell_deck_load(self, data);
|
||||
}
|
||||
|
||||
pub fn clone(&self) -> &'static mut Self {
|
||||
import!(logic_spell_deck_clone(ptr: *const LogicSpellDeck) -> *mut LogicSpellDeck = 0x11890C + 1);
|
||||
unsafe { &mut *logic_spell_deck_clone(self) }
|
||||
}
|
||||
}
|
||||
|
||||
impl LogicSpellCollection {
|
||||
pub fn add_spell(&mut self, spell: &'static LogicSpell) {
|
||||
import!(logic_spell_collection_add_spell(ptr: *const LogicSpellCollection, spell: *const LogicSpell) -> () = 0x118708 + 1);
|
||||
logic_spell_collection_add_spell(self, spell);
|
||||
}
|
||||
}
|
9
libserver/src/logic/time.rs
Normal file
9
libserver/src/logic/time.rs
Normal file
|
@ -0,0 +1,9 @@
|
|||
#[repr(C)]
|
||||
pub struct LogicTime {
|
||||
pub tick: i32, // 0
|
||||
pub max_tick: i32, // 4
|
||||
pub time_multiplier: i32, // 8
|
||||
pub real_time: i32, // 12
|
||||
pub prev_tick: i32, // (?) 16
|
||||
unk_gap_1: [u8; 40], // 20 - 60
|
||||
}
|
8
libserver/src/network/connection.rs
Normal file
8
libserver/src/network/connection.rs
Normal file
|
@ -0,0 +1,8 @@
|
|||
#[repr(C)]
|
||||
pub struct Connection {
|
||||
pub fd: i32,
|
||||
pub is_connected: bool,
|
||||
unk_1: i32,
|
||||
unk_2: i32,
|
||||
unk_3: i32,
|
||||
}
|
18
libserver/src/network/encryption.rs
Normal file
18
libserver/src/network/encryption.rs
Normal file
|
@ -0,0 +1,18 @@
|
|||
use crate::{allocator, ffi_util::import, sc_string::ScString};
|
||||
|
||||
#[repr(C)]
|
||||
pub struct RC4Encrypter {
|
||||
vtable: usize,
|
||||
x: i32,
|
||||
y: i32,
|
||||
key: [u8; 256],
|
||||
}
|
||||
|
||||
impl RC4Encrypter {
|
||||
pub fn new(key: &str, nonce: &str) -> *const Self {
|
||||
import!(rc4_encrypter_ctor(ptr: *const RC4Encrypter, key: *const ScString, nonce: *const ScString) -> () = 0x15C1B0 + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
rc4_encrypter_ctor(instance, ScString::new(key), ScString::new(nonce));
|
||||
instance
|
||||
}
|
||||
}
|
77
libserver/src/network/message/account.rs
Normal file
77
libserver/src/network/message/account.rs
Normal file
|
@ -0,0 +1,77 @@
|
|||
use crate::{
|
||||
allocator,
|
||||
ffi_util::{import, Nullable},
|
||||
logic::math::LogicLong,
|
||||
sc_string::ScString,
|
||||
};
|
||||
|
||||
use super::PiranhaMessage;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LoginMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
pub account_id: &'static LogicLong,
|
||||
pub pass_token: Nullable<ScString>,
|
||||
pub client_major_version: i32,
|
||||
pub client_minor_version: i32,
|
||||
pub client_build: i32,
|
||||
}
|
||||
|
||||
impl LoginMessage {
|
||||
pub const MESSAGE_TYPE: u16 = 10101;
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct KeepAliveMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
}
|
||||
|
||||
impl KeepAliveMessage {
|
||||
pub const MESSAGE_TYPE: u16 = 10108;
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LoginOkMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
pub account_id: &'static LogicLong,
|
||||
pub home_id: &'static LogicLong,
|
||||
pub pass_token: Nullable<ScString>,
|
||||
pub facebook_id: Nullable<ScString>,
|
||||
pub gamecenter_id: Nullable<ScString>,
|
||||
pub facebook_app_id: Nullable<ScString>,
|
||||
pub server_major_version: i32,
|
||||
pub server_minor_version: i32,
|
||||
pub server_build: i32,
|
||||
pub content_version: i32,
|
||||
pub environment: Nullable<ScString>,
|
||||
pub session_count: i32,
|
||||
pub play_time_seconds: i32,
|
||||
pub days_since_started_playing: i32,
|
||||
pub server_time: Nullable<ScString>,
|
||||
pub account_created_date: Nullable<ScString>,
|
||||
pub google_service_id: Nullable<ScString>,
|
||||
pub unk: i32,
|
||||
}
|
||||
|
||||
impl LoginOkMessage {
|
||||
pub fn new() -> &'static mut Self {
|
||||
import!(login_ok_message_ctor(ptr: *mut LoginOkMessage) -> () = 0x12A644 + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
login_ok_message_ctor(instance);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct KeepAliveServerMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
}
|
||||
|
||||
impl KeepAliveServerMessage {
|
||||
pub fn new() -> &'static mut Self {
|
||||
import!(keep_alive_server_message_ctor(ptr: *mut KeepAliveServerMessage) -> () = 0x129E7C + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
keep_alive_server_message_ctor(instance);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
}
|
76
libserver/src/network/message/home.rs
Normal file
76
libserver/src/network/message/home.rs
Normal file
|
@ -0,0 +1,76 @@
|
|||
use crate::{
|
||||
allocator,
|
||||
array_list::LogicArrayList,
|
||||
ffi_util::{import, Nullable},
|
||||
logic::{
|
||||
avatar::LogicClientAvatar, command::LogicCommand, data::LogicNpcData, home::LogicClientHome,
|
||||
},
|
||||
};
|
||||
|
||||
use super::PiranhaMessage;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct GoHomeMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
unk_bool: bool,
|
||||
}
|
||||
|
||||
impl GoHomeMessage {
|
||||
pub const MESSAGE_TYPE: u16 = 14101;
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct EndClientTurnMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
pub commands: &'static LogicArrayList<&'static LogicCommand>,
|
||||
pub tick: i32,
|
||||
pub checksum: i32,
|
||||
}
|
||||
|
||||
impl EndClientTurnMessage {
|
||||
pub const MESSAGE_TYPE: u16 = 14102;
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct StartMissionMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
pub npc: &'static LogicNpcData,
|
||||
}
|
||||
|
||||
impl StartMissionMessage {
|
||||
pub const MESSAGE_TYPE: u16 = 14104;
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct OwnHomeDataMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
pub logic_client_home: Nullable<LogicClientHome>,
|
||||
pub logic_client_avatar: Nullable<LogicClientAvatar>,
|
||||
pub random_seed: i32,
|
||||
}
|
||||
|
||||
impl OwnHomeDataMessage {
|
||||
pub fn new() -> &'static mut Self {
|
||||
import!(own_home_data_message_ctor(ptr: *mut OwnHomeDataMessage) -> () = 0x130480 + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
own_home_data_message_ctor(instance);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct OutOfSyncMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
pub server_checksum: i32,
|
||||
pub client_checksum: i32,
|
||||
pub tick: i32,
|
||||
}
|
||||
|
||||
impl OutOfSyncMessage {
|
||||
pub fn new() -> &'static mut Self {
|
||||
import!(out_of_sync_message_ctor(ptr: *mut OutOfSyncMessage) -> () = 0x130330 + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
out_of_sync_message_ctor(instance);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
}
|
27
libserver/src/network/message/mod.rs
Normal file
27
libserver/src/network/message/mod.rs
Normal file
|
@ -0,0 +1,27 @@
|
|||
mod account;
|
||||
mod home;
|
||||
mod sector;
|
||||
|
||||
pub use account::*;
|
||||
pub use home::*;
|
||||
pub use sector::*;
|
||||
|
||||
use crate::util::data_stream::ByteStream;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct PiranhaMessageVTable {
|
||||
finalizer_1: extern "C" fn(ptr: *const PiranhaMessage),
|
||||
finalizer_2: extern "C" fn(ptr: *const PiranhaMessage),
|
||||
encode: extern "C" fn(ptr: *const PiranhaMessage),
|
||||
decode: extern "C" fn(ptr: *const PiranhaMessage),
|
||||
pub get_service_node_type: extern "C" fn() -> i32,
|
||||
pub get_message_type: extern "C" fn() -> u16,
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct PiranhaMessage {
|
||||
pub vtable: &'static PiranhaMessageVTable,
|
||||
pub version: i32,
|
||||
pub stream: ByteStream,
|
||||
pub proxy_session_id: i32,
|
||||
}
|
17
libserver/src/network/message/sector.rs
Normal file
17
libserver/src/network/message/sector.rs
Normal file
|
@ -0,0 +1,17 @@
|
|||
use crate::{allocator, ffi_util::import};
|
||||
|
||||
use super::PiranhaMessage;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct SectorStateMessage {
|
||||
pub parent: PiranhaMessage,
|
||||
}
|
||||
|
||||
impl SectorStateMessage {
|
||||
pub fn new() -> &'static mut Self {
|
||||
import!(sector_state_message_ctor(ptr: *mut SectorStateMessage) -> () = 0x132DA0 + 1);
|
||||
let instance = allocator::new::<Self>();
|
||||
sector_state_message_ctor(instance);
|
||||
unsafe { &mut *instance }
|
||||
}
|
||||
}
|
23
libserver/src/network/message_factory.rs
Normal file
23
libserver/src/network/message_factory.rs
Normal file
|
@ -0,0 +1,23 @@
|
|||
use std::mem::MaybeUninit;
|
||||
|
||||
use crate::ffi_util::import;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct LogicScrollMessageFactory {
|
||||
vtable: usize,
|
||||
is_udp_factory: bool,
|
||||
}
|
||||
|
||||
impl LogicScrollMessageFactory {
|
||||
pub const RC4_KEY: &str = "fhsd6f86f67rt8fw78fw789we78r9789wer6re";
|
||||
|
||||
pub fn new(is_udp_factory: bool) -> Self {
|
||||
import!(logic_scroll_message_factory_ctor(ptr: *const LogicScrollMessageFactory, is_udp: bool) -> () = 0x118F60 + 1);
|
||||
|
||||
unsafe {
|
||||
let instance = MaybeUninit::zeroed().assume_init();
|
||||
logic_scroll_message_factory_ctor(&instance, is_udp_factory);
|
||||
instance
|
||||
}
|
||||
}
|
||||
}
|
57
libserver/src/network/messaging.rs
Normal file
57
libserver/src/network/messaging.rs
Normal file
|
@ -0,0 +1,57 @@
|
|||
use crate::{allocator::malloc, ffi_util::import};
|
||||
|
||||
use super::{message::PiranhaMessage, Connection, LogicScrollMessageFactory, RC4Encrypter};
|
||||
|
||||
#[repr(transparent)]
|
||||
pub struct Messaging(pub *const u8);
|
||||
|
||||
impl Messaging {
|
||||
pub fn new(queue_size: usize) -> Self {
|
||||
import!(messaging_ctor(ptr: *const u8, queue_size: i32) -> () = 0x138C2C + 1);
|
||||
let instance = malloc(300);
|
||||
messaging_ctor(instance, queue_size as i32);
|
||||
Self(instance)
|
||||
}
|
||||
|
||||
pub fn set_encrypters(
|
||||
&mut self,
|
||||
encrypter: *const RC4Encrypter,
|
||||
decrypter: *const RC4Encrypter,
|
||||
) {
|
||||
import!(messaging_set_encrypters(ptr: *const u8, en: *const RC4Encrypter, de: *const RC4Encrypter, a4: bool) -> () = 0x138BFC + 1);
|
||||
messaging_set_encrypters(self.0, encrypter, decrypter, false);
|
||||
}
|
||||
|
||||
pub fn on_receive(&self, connection: &Connection) {
|
||||
import!(messaging_on_receive(ptr: *const u8, connection: *const Connection) -> () = 0x139650 + 1);
|
||||
messaging_on_receive(self.0, connection);
|
||||
}
|
||||
|
||||
pub fn next_message(&self) -> Option<&'static PiranhaMessage> {
|
||||
import!(messaging_next_message(ptr: *const u8) -> *const PiranhaMessage = 0x1390B4 + 1);
|
||||
let message = messaging_next_message(self.0);
|
||||
(!message.is_null()).then(|| unsafe { &*message })
|
||||
}
|
||||
|
||||
pub fn send<T>(&self, message: *const T) {
|
||||
import!(messaging_send(ptr: *const u8, message: usize) -> () = 0x138FB8 + 1);
|
||||
messaging_send(self.0, message as usize);
|
||||
}
|
||||
|
||||
pub fn on_wakeup(&self, connection: &Connection) {
|
||||
import!(messaging_on_wakeup(ptr: *const u8, connection: *const Connection) -> () = 0x1394C4 + 1);
|
||||
messaging_on_wakeup(self.0, connection);
|
||||
}
|
||||
|
||||
pub fn get_connection(&self) -> &Connection {
|
||||
unsafe { &*(self.0.wrapping_add(64) as *const Connection) }
|
||||
}
|
||||
|
||||
pub fn get_connection_mut(&self) -> &mut Connection {
|
||||
unsafe { &mut *(self.0.wrapping_add(64) as *mut Connection) }
|
||||
}
|
||||
|
||||
pub fn set_message_factory(&mut self, factory: &'static LogicScrollMessageFactory) {
|
||||
unsafe { *(self.0.wrapping_add(4) as *mut *const LogicScrollMessageFactory) = factory }
|
||||
}
|
||||
}
|
11
libserver/src/network/mod.rs
Normal file
11
libserver/src/network/mod.rs
Normal file
|
@ -0,0 +1,11 @@
|
|||
mod connection;
|
||||
mod encryption;
|
||||
mod message_factory;
|
||||
mod messaging;
|
||||
|
||||
pub mod message;
|
||||
|
||||
pub use connection::Connection;
|
||||
pub use encryption::RC4Encrypter;
|
||||
pub use message_factory::LogicScrollMessageFactory;
|
||||
pub use messaging::Messaging;
|
142
libserver/src/resources.rs
Normal file
142
libserver/src/resources.rs
Normal file
|
@ -0,0 +1,142 @@
|
|||
use crate::{
|
||||
allocator::malloc,
|
||||
array_list::LogicArrayList,
|
||||
ffi_util::import,
|
||||
logic::data::{LogicDataTables, LogicLocationData},
|
||||
sc_string::ScString,
|
||||
};
|
||||
|
||||
#[repr(transparent)]
|
||||
pub struct DataLoaderFactory(*const u8);
|
||||
|
||||
impl DataLoaderFactory {
|
||||
pub fn new() -> Self {
|
||||
import!(data_loader_factory_ctor(ptr: *const u8) -> () = 0x15EC38 + 1);
|
||||
|
||||
let instance = malloc(4);
|
||||
data_loader_factory_ctor(instance);
|
||||
Self(instance)
|
||||
}
|
||||
}
|
||||
|
||||
struct LogicResources;
|
||||
|
||||
impl LogicResources {
|
||||
pub fn create_data_table_resources_array() -> &'static LogicArrayList<LogicDataTableResource> {
|
||||
import!(logic_resources_create_data_table_resources_array() -> *const u8 = 0x1078E8 + 1);
|
||||
unsafe {
|
||||
&*(logic_resources_create_data_table_resources_array()
|
||||
as *const LogicArrayList<LogicDataTableResource>)
|
||||
}
|
||||
}
|
||||
|
||||
pub fn load(
|
||||
resources_array: &LogicArrayList<LogicDataTableResource>,
|
||||
index: usize,
|
||||
csv: &CSVNode,
|
||||
) {
|
||||
import!(logic_resources_load(r: *const LogicArrayList<LogicDataTableResource>, i: i32, csv: *const u8, a4: i32, a5: i32) -> () = 0x10833C + 1);
|
||||
logic_resources_load(
|
||||
resources_array as *const LogicArrayList<LogicDataTableResource>,
|
||||
index as i32,
|
||||
csv.0,
|
||||
0,
|
||||
0,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
#[repr(transparent)]
|
||||
struct LogicDataTableResource(*const u8);
|
||||
|
||||
impl LogicDataTableResource {
|
||||
pub fn get_file_name(&self) -> String {
|
||||
unsafe { (&*(self.0.wrapping_add(8) as *const ScString)).to_string() }
|
||||
}
|
||||
}
|
||||
|
||||
#[repr(transparent)]
|
||||
pub struct CSVNode(pub *const u8);
|
||||
|
||||
pub struct ResourceManager;
|
||||
|
||||
impl ResourceManager {
|
||||
pub fn init(data_loader_factory: &DataLoaderFactory) {
|
||||
import!(resource_manager_init(dlf: *const u8, data_path: *const u8) -> () = 0x15D668 + 1);
|
||||
resource_manager_init(data_loader_factory.0, [0x00].as_ptr());
|
||||
}
|
||||
|
||||
pub fn resource_to_load() -> i32 {
|
||||
import!(resource_manager_resource_to_load() -> i32 = 0x15E068 + 1);
|
||||
resource_manager_resource_to_load()
|
||||
}
|
||||
|
||||
pub fn load_next_resource() {
|
||||
import!(resource_manager_load_next_resource() -> () = 0x15DC10 + 1);
|
||||
resource_manager_load_next_resource();
|
||||
}
|
||||
|
||||
pub fn get_csv(file_name: &str) -> CSVNode {
|
||||
import!(resource_manager_get_csv(file_name: *const ScString) -> *const u8 = 0x15E08C + 1);
|
||||
CSVNode(resource_manager_get_csv(ScString::new(file_name)))
|
||||
}
|
||||
}
|
||||
|
||||
#[repr(transparent)]
|
||||
struct ResourceListener(*const u8);
|
||||
|
||||
impl ResourceListener {
|
||||
pub fn new() -> Self {
|
||||
import!(resource_listener_ctor(ptr: *const u8) -> () = 0x15C370 + 1);
|
||||
|
||||
let instance = malloc(20);
|
||||
resource_listener_ctor(instance);
|
||||
Self(instance)
|
||||
}
|
||||
|
||||
pub fn add_file(&self, name: &str) {
|
||||
import!(resource_listener_add_file(ptr: *const u8, name: *const ScString, a3: i32, a4: i32, a5: i32, a6: i32) -> () = 0x15C51C + 1);
|
||||
resource_listener_add_file(self.0, ScString::new(name), -1, -1, -1, -1);
|
||||
}
|
||||
|
||||
pub fn start_loading(&self) {
|
||||
import!(resource_listener_start_loading(ptr: *const u8) -> () = 0x15C7BC + 1);
|
||||
resource_listener_start_loading(self.0);
|
||||
}
|
||||
}
|
||||
|
||||
pub fn init() {
|
||||
let data_loader_factory = DataLoaderFactory::new();
|
||||
ResourceManager::init(&data_loader_factory);
|
||||
|
||||
let listener = ResourceListener::new();
|
||||
let data_table_resources_array = LogicResources::create_data_table_resources_array();
|
||||
|
||||
for resource in data_table_resources_array.as_slice().iter() {
|
||||
listener.add_file(&resource.get_file_name());
|
||||
}
|
||||
|
||||
listener.start_loading();
|
||||
|
||||
while ResourceManager::resource_to_load() != 0 {
|
||||
ResourceManager::load_next_resource();
|
||||
}
|
||||
|
||||
for (index, resource) in data_table_resources_array.as_slice().iter().enumerate() {
|
||||
let csv = ResourceManager::get_csv(&resource.get_file_name());
|
||||
LogicResources::load(data_table_resources_array, index, &csv);
|
||||
}
|
||||
}
|
||||
|
||||
pub fn prepare_location_data(location: &LogicLocationData) {
|
||||
let tilemap_file_name = location.get_tilemap_file_name();
|
||||
load_one_resource(&tilemap_file_name);
|
||||
|
||||
let csv = ResourceManager::get_csv(&tilemap_file_name);
|
||||
LogicDataTables::init_tilemap(csv, location.parent.global_id % 1_000_000);
|
||||
}
|
||||
|
||||
pub fn load_one_resource(path: &str) {
|
||||
import!(game_main_load_asset(path: *const ScString) -> () = 0x5A73C + 1);
|
||||
game_main_load_asset(ScString::new(path));
|
||||
}
|
72
libserver/src/sc_string.rs
Normal file
72
libserver/src/sc_string.rs
Normal file
|
@ -0,0 +1,72 @@
|
|||
use std::{
|
||||
ffi::{CStr, CString},
|
||||
fmt,
|
||||
};
|
||||
|
||||
use crate::{allocator, ffi_util::import};
|
||||
|
||||
#[repr(C)]
|
||||
pub struct ScString {
|
||||
unk_1: i32,
|
||||
length: i32,
|
||||
content: [u8; 8],
|
||||
unk_2: i32,
|
||||
}
|
||||
|
||||
impl<S> From<S> for &'static ScString
|
||||
where
|
||||
S: AsRef<str>,
|
||||
{
|
||||
fn from(value: S) -> Self {
|
||||
import!(string_ctor(ptr: *const ScString, data: *const u8) -> () = 0x13AF4C + 1);
|
||||
let sc_string = unsafe { &*allocator::new::<ScString>() };
|
||||
|
||||
string_ctor(sc_string, CString::new(value.as_ref()).unwrap().as_ptr());
|
||||
sc_string
|
||||
}
|
||||
}
|
||||
|
||||
impl ScString {
|
||||
pub fn new<S: AsRef<str>>(value: S) -> &'static Self {
|
||||
From::from(value)
|
||||
}
|
||||
|
||||
pub fn assign(&mut self, content: &str) {
|
||||
import!(string_assignment_operator(lhs: *mut ScString, rhs: *const ScString) -> () = 0x13B16C + 1);
|
||||
string_assignment_operator(self, ScString::new(content));
|
||||
}
|
||||
}
|
||||
|
||||
impl fmt::Display for ScString {
|
||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||
let length = self.length as usize;
|
||||
if length + 1 > 8 {
|
||||
let c_string = usize::from_le_bytes(self.content[..4].try_into().unwrap()) as *const u8;
|
||||
unsafe {
|
||||
write!(
|
||||
f,
|
||||
"{}",
|
||||
CStr::from_ptr(c_string).to_string_lossy().to_string()
|
||||
)
|
||||
}
|
||||
} else if length > 0 {
|
||||
unsafe {
|
||||
write!(
|
||||
f,
|
||||
"{}",
|
||||
CStr::from_ptr(self.content.as_ptr())
|
||||
.to_string_lossy()
|
||||
.to_string()
|
||||
)
|
||||
}
|
||||
} else {
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl fmt::Debug for ScString {
|
||||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||
write!(f, "\"{}\"", self)
|
||||
}
|
||||
}
|
51
libserver/src/session/handlers/account.rs
Normal file
51
libserver/src/session/handlers/account.rs
Normal file
|
@ -0,0 +1,51 @@
|
|||
use tracing::info;
|
||||
|
||||
use crate::{
|
||||
logic::{
|
||||
math::LogicLong,
|
||||
LogicVersion,
|
||||
},
|
||||
network::message::{
|
||||
KeepAliveMessage, KeepAliveServerMessage, LoginMessage, LoginOkMessage,
|
||||
},
|
||||
sc_string::ScString,
|
||||
session::{PlayerSaveData, PlayerSession},
|
||||
};
|
||||
|
||||
pub fn on_login_message(session: &mut PlayerSession, message: &LoginMessage) {
|
||||
info!(
|
||||
"LoginMessage received: account_id: {}, pass_token: {:?}, client version: {}.{}.{}",
|
||||
message.account_id,
|
||||
message.pass_token.get(),
|
||||
message.client_major_version,
|
||||
message.client_build,
|
||||
message.client_minor_version
|
||||
);
|
||||
|
||||
let login_ok_message = LoginOkMessage::new();
|
||||
|
||||
login_ok_message.account_id = LogicLong::new(0, 1);
|
||||
login_ok_message.home_id = LogicLong::new(0, 1);
|
||||
login_ok_message
|
||||
.pass_token
|
||||
.set(ScString::new("MostSecureTokenEver"));
|
||||
|
||||
login_ok_message.server_major_version = LogicVersion::MAJOR;
|
||||
login_ok_message.server_build = LogicVersion::BUILD;
|
||||
login_ok_message.environment.set(ScString::new("dev"));
|
||||
|
||||
let account_id = LogicLong {
|
||||
higher_int: 0,
|
||||
lower_int: 1,
|
||||
};
|
||||
|
||||
session.player_save_data = Some(PlayerSaveData::create_default_data(&account_id));
|
||||
session.account_id = Some(account_id);
|
||||
|
||||
session.send_message(login_ok_message);
|
||||
session.go_home();
|
||||
}
|
||||
|
||||
pub fn on_keep_alive_message(session: &mut PlayerSession, _message: &KeepAliveMessage) {
|
||||
session.send_message(KeepAliveServerMessage::new());
|
||||
}
|
71
libserver/src/session/handlers/home.rs
Normal file
71
libserver/src/session/handlers/home.rs
Normal file
|
@ -0,0 +1,71 @@
|
|||
use tracing::{error, info};
|
||||
|
||||
use crate::{
|
||||
network::message::{
|
||||
EndClientTurnMessage, GoHomeMessage, OutOfSyncMessage,
|
||||
StartMissionMessage,
|
||||
},
|
||||
session::{PlayerSaveData, PlayerSession},
|
||||
};
|
||||
|
||||
pub fn on_go_home_message(session: &mut PlayerSession, _message: &GoHomeMessage) {
|
||||
session.go_home();
|
||||
}
|
||||
|
||||
pub fn on_end_client_turn_message(session: &mut PlayerSession, message: &EndClientTurnMessage) {
|
||||
let Some(logic_game_mode) = session.logic_game_mode.as_mut() else {
|
||||
error!("EndClientTurnMessage received but LogicGameMode is NULL!");
|
||||
return;
|
||||
};
|
||||
|
||||
info!(
|
||||
"EndClientTurnMessage received: tick: {}, checksum: {}, command count: {}",
|
||||
message.tick, message.checksum, message.commands.count
|
||||
);
|
||||
|
||||
while logic_game_mode.time.tick < message.tick {
|
||||
for command in message.commands.as_slice() {
|
||||
if command.execute_tick == logic_game_mode.time.tick {
|
||||
info!(
|
||||
"received command of type: {}",
|
||||
(command.vtable.get_command_type)()
|
||||
);
|
||||
logic_game_mode.logic_command_manager.add_command(command);
|
||||
}
|
||||
}
|
||||
|
||||
logic_game_mode.update_one_tick();
|
||||
}
|
||||
|
||||
let checksum = logic_game_mode.calculate_checksum();
|
||||
if checksum != message.checksum {
|
||||
error!(
|
||||
"client and server are out of sync! tick: {}, server checksum: {}, client checksum: {}",
|
||||
logic_game_mode.time.tick, checksum, message.checksum
|
||||
);
|
||||
|
||||
let out_of_sync_message = OutOfSyncMessage::new();
|
||||
out_of_sync_message.server_checksum = checksum;
|
||||
out_of_sync_message.client_checksum = message.checksum;
|
||||
out_of_sync_message.tick = logic_game_mode.time.tick;
|
||||
session.messaging.send(out_of_sync_message);
|
||||
}
|
||||
|
||||
if let Some(logic_client_home) = logic_game_mode.home.get() {
|
||||
// Save data if we're in home state
|
||||
|
||||
session.player_save_data = Some(PlayerSaveData::new(
|
||||
logic_client_home,
|
||||
logic_game_mode.player_avatar.get().unwrap(),
|
||||
));
|
||||
}
|
||||
}
|
||||
|
||||
pub fn on_start_mission_message(session: &mut PlayerSession, message: &StartMissionMessage) {
|
||||
info!(
|
||||
"StartMissionMessage received: npc: {}, location: {}",
|
||||
message.npc.parent.global_id, message.npc.location.parent.global_id
|
||||
);
|
||||
|
||||
session.start_mission(message.npc);
|
||||
}
|
30
libserver/src/session/handlers/mod.rs
Normal file
30
libserver/src/session/handlers/mod.rs
Normal file
|
@ -0,0 +1,30 @@
|
|||
mod account;
|
||||
mod home;
|
||||
|
||||
use account::*;
|
||||
use home::*;
|
||||
|
||||
macro_rules! messages {
|
||||
($($name:ident),*) => {
|
||||
pub fn handle_message(
|
||||
session: &mut super::PlayerSession,
|
||||
message: &crate::network::message::PiranhaMessage,
|
||||
) {
|
||||
use crate::network::message::*;
|
||||
|
||||
match (message.vtable.get_message_type)() {
|
||||
$(
|
||||
$name::MESSAGE_TYPE => ::paste::paste!([<on_ $name:snake>](
|
||||
session,
|
||||
unsafe { ::std::mem::transmute(message) }
|
||||
)),
|
||||
)*
|
||||
unhandled => ::tracing::warn!("unhandled message: {unhandled}"),
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
messages! {
|
||||
LoginMessage, KeepAliveMessage, GoHomeMessage, EndClientTurnMessage, StartMissionMessage
|
||||
}
|
118
libserver/src/session/mod.rs
Normal file
118
libserver/src/session/mod.rs
Normal file
|
@ -0,0 +1,118 @@
|
|||
use rand::RngCore;
|
||||
use save::PlayerSaveData;
|
||||
|
||||
use crate::{
|
||||
allocator,
|
||||
logic::{
|
||||
avatar::LogicClientAvatar,
|
||||
data::{LogicDataTables, LogicNpcData},
|
||||
math::LogicLong,
|
||||
mode::LogicGameMode,
|
||||
},
|
||||
network::{message::PiranhaMessage, Messaging},
|
||||
resources,
|
||||
};
|
||||
|
||||
mod handlers;
|
||||
mod protocol_util;
|
||||
mod save;
|
||||
|
||||
pub struct PlayerSession {
|
||||
messaging: Messaging,
|
||||
pub account_id: Option<LogicLong>,
|
||||
pub player_save_data: Option<PlayerSaveData>,
|
||||
pub logic_game_mode: Option<&'static mut LogicGameMode>,
|
||||
}
|
||||
|
||||
impl PlayerSession {
|
||||
pub fn new(messaging: Messaging) -> Self {
|
||||
Self {
|
||||
messaging,
|
||||
account_id: None,
|
||||
player_save_data: None,
|
||||
logic_game_mode: None,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn send_message<T>(&self, message: *const T) {
|
||||
self.messaging.send(message);
|
||||
}
|
||||
|
||||
pub fn go_home(&mut self) {
|
||||
self.destruct_game_mode();
|
||||
|
||||
let player_save_data = self.player_save_data.as_ref().unwrap();
|
||||
|
||||
let logic_client_home = player_save_data.get_logic_client_home();
|
||||
let logic_client_avatar = player_save_data.get_logic_client_avatar();
|
||||
let random_seed = rand::rng().next_u32() as i32;
|
||||
|
||||
let own_home_data_message =
|
||||
protocol_util::build_own_home_data(logic_client_home, logic_client_avatar, random_seed);
|
||||
|
||||
let logic_game_mode = LogicGameMode::new(false);
|
||||
logic_game_mode.load_home_state(logic_client_home, logic_client_avatar, 0, -1, random_seed);
|
||||
|
||||
self.logic_game_mode = Some(logic_game_mode);
|
||||
self.send_message(own_home_data_message);
|
||||
}
|
||||
|
||||
pub fn start_mission(&mut self, npc: &'static LogicNpcData) {
|
||||
const TRAINING_ARENA_ID: i32 = 54000001;
|
||||
|
||||
self.destruct_game_mode();
|
||||
|
||||
resources::prepare_location_data(npc.location);
|
||||
let arena = LogicDataTables::get_data_by_id(TRAINING_ARENA_ID).unwrap();
|
||||
let player_save_data = self.player_save_data.as_ref().unwrap();
|
||||
let player_avatar = player_save_data.get_logic_client_avatar();
|
||||
|
||||
let player_deck = player_save_data.get_spell_deck();
|
||||
let npc_deck = player_deck.clone();
|
||||
|
||||
let npc_avatar = LogicClientAvatar::new();
|
||||
npc_avatar.id.set(-1, -1);
|
||||
npc_avatar.account_id.set(-1, -1);
|
||||
npc_avatar.home_id.set(-1, -1);
|
||||
npc_avatar.arena = arena;
|
||||
npc_avatar.level = player_avatar.level;
|
||||
|
||||
let battle_mode = LogicGameMode::new(true);
|
||||
let battle = battle_mode.battle.get_mut().unwrap();
|
||||
|
||||
battle.set_location(npc.location, false, 0);
|
||||
battle.battle_type = 1;
|
||||
battle.npc_data = npc;
|
||||
battle.arena_data = arena;
|
||||
|
||||
battle.set_spell_decks(player_deck, npc_deck);
|
||||
battle_mode.add_player(player_avatar);
|
||||
battle_mode.add_player(npc_avatar);
|
||||
|
||||
let sector_state_message = protocol_util::build_sector_state(battle_mode, player_avatar);
|
||||
|
||||
self.logic_game_mode = Some(battle_mode);
|
||||
self.send_message(sector_state_message);
|
||||
}
|
||||
|
||||
pub fn destruct_game_mode(&mut self) {
|
||||
if let Some(logic_game_mode) = self.logic_game_mode.take() {
|
||||
logic_game_mode.destruct();
|
||||
allocator::free(logic_game_mode);
|
||||
}
|
||||
}
|
||||
|
||||
pub fn work_until_disconnect(&mut self) {
|
||||
while self.messaging.get_connection().is_connected {
|
||||
self.messaging.on_receive(self.messaging.get_connection());
|
||||
while let Some(message) = self.messaging.next_message() {
|
||||
self.receive_message(message);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn receive_message(&mut self, message: &PiranhaMessage) {
|
||||
handlers::handle_message(self, message);
|
||||
self.messaging.on_wakeup(self.messaging.get_connection());
|
||||
}
|
||||
}
|
40
libserver/src/session/protocol_util.rs
Normal file
40
libserver/src/session/protocol_util.rs
Normal file
|
@ -0,0 +1,40 @@
|
|||
use crate::{
|
||||
logic::{avatar::LogicClientAvatar, home::LogicClientHome, mode::LogicGameMode},
|
||||
network::message::{OwnHomeDataMessage, SectorStateMessage},
|
||||
};
|
||||
|
||||
pub fn build_own_home_data(
|
||||
logic_client_home: &LogicClientHome,
|
||||
logic_client_avatar: &LogicClientAvatar,
|
||||
random_seed: i32,
|
||||
) -> &'static mut OwnHomeDataMessage {
|
||||
let own_home_data_message = OwnHomeDataMessage::new();
|
||||
|
||||
own_home_data_message
|
||||
.logic_client_home
|
||||
.set(LogicClientHome::clone(logic_client_home));
|
||||
|
||||
own_home_data_message
|
||||
.logic_client_avatar
|
||||
.set(LogicClientAvatar::clone(logic_client_avatar));
|
||||
|
||||
own_home_data_message.random_seed = random_seed;
|
||||
own_home_data_message
|
||||
}
|
||||
|
||||
pub fn build_sector_state(
|
||||
logic_game_mode: &LogicGameMode,
|
||||
client_avatar: &LogicClientAvatar,
|
||||
) -> &'static mut SectorStateMessage {
|
||||
let sector_state_message = SectorStateMessage::new();
|
||||
sector_state_message.parent.stream.write_byte(0);
|
||||
sector_state_message.parent.stream.parent.checksum = 0;
|
||||
logic_game_mode.encode(
|
||||
&mut sector_state_message.parent.stream.parent,
|
||||
true,
|
||||
client_avatar,
|
||||
false,
|
||||
);
|
||||
|
||||
sector_state_message
|
||||
}
|
112
libserver/src/session/save.rs
Normal file
112
libserver/src/session/save.rs
Normal file
|
@ -0,0 +1,112 @@
|
|||
use crate::{
|
||||
allocator,
|
||||
logic::{
|
||||
avatar::LogicClientAvatar,
|
||||
data::LogicDataTables,
|
||||
home::LogicClientHome,
|
||||
json::LogicJSONParser,
|
||||
math::LogicLong,
|
||||
spell::{LogicSpell, LogicSpellDeck},
|
||||
},
|
||||
util::data_stream::ByteStream,
|
||||
};
|
||||
|
||||
pub struct PlayerSaveData {
|
||||
pub logic_client_home: Vec<u8>,
|
||||
pub logic_client_avatar: Vec<u8>,
|
||||
}
|
||||
|
||||
impl PlayerSaveData {
|
||||
pub fn new(
|
||||
logic_client_home: &LogicClientHome,
|
||||
logic_client_avatar: &LogicClientAvatar,
|
||||
) -> Self {
|
||||
let mut stream = ByteStream::new(64);
|
||||
logic_client_home.encode(&mut stream.parent);
|
||||
let logic_client_home = stream.get_data().to_vec();
|
||||
stream.destruct();
|
||||
|
||||
let mut stream = ByteStream::new(64);
|
||||
(logic_client_avatar.vtable.encode)(logic_client_avatar, &mut stream.parent);
|
||||
let logic_client_avatar = stream.get_data().to_vec();
|
||||
stream.destruct();
|
||||
|
||||
Self {
|
||||
logic_client_home,
|
||||
logic_client_avatar,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn create_default_data(id: &LogicLong) -> Self {
|
||||
const STARTING_DECK_JSON: &str = r#"[{"d":26000000,"t":0,"s":0,"l":12,"c":1},{"d":26000001,"t":0,"s":0,"l":12,"c":1},{"d":26000013,"t":0,"s":0,"l":12,"c":1},{"d":28000001,"t":0,"s":0,"l":12,"c":1},{"d":28000000,"t":0,"s":0,"l":10,"c":1},{"d":26000003,"t":0,"s":0,"l":10,"c":1},{"d":26000004,"t":0,"s":0,"l":7,"c":1},{"d":26000005,"t":0,"s":0,"l":12,"c":1}]"#;
|
||||
|
||||
let logic_client_home = LogicClientHome::new();
|
||||
logic_client_home
|
||||
.id
|
||||
.set(LogicLong::new(id.higher_int, id.lower_int));
|
||||
|
||||
let starting_deck = LogicJSONParser::parse(STARTING_DECK_JSON).unwrap();
|
||||
logic_client_home.spell_deck.load(starting_deck);
|
||||
|
||||
for i in 0..LogicDataTables::get_spell_count() {
|
||||
let spell_data = LogicDataTables::get_spell_at(i);
|
||||
if !logic_client_home.has_spell(spell_data) {
|
||||
let spell = LogicSpell::new();
|
||||
spell.data = spell_data;
|
||||
spell.level_index = spell_data.get_max_level_index();
|
||||
logic_client_home.spell_collection.add_spell(spell);
|
||||
}
|
||||
}
|
||||
|
||||
let logic_client_avatar = LogicClientAvatar::new();
|
||||
logic_client_avatar.id.set(id.higher_int, id.lower_int);
|
||||
logic_client_avatar
|
||||
.account_id
|
||||
.set(id.higher_int, id.lower_int);
|
||||
logic_client_avatar.home_id.set(id.higher_int, id.lower_int);
|
||||
logic_client_avatar.name.assign("xeondev");
|
||||
logic_client_avatar.name_change_state = 0;
|
||||
logic_client_avatar.name_set_by_user = true;
|
||||
logic_client_avatar.arena = LogicDataTables::get_data_by_id(54000007).unwrap();
|
||||
logic_client_avatar.npc_win_count = 8;
|
||||
logic_client_avatar.score = 3000;
|
||||
logic_client_avatar.level = 13;
|
||||
|
||||
let instance = Self::new(logic_client_home, logic_client_avatar);
|
||||
|
||||
logic_client_home.destruct();
|
||||
allocator::free(logic_client_home as *const _);
|
||||
|
||||
(logic_client_avatar.vtable.destruct)(logic_client_avatar);
|
||||
allocator::free(logic_client_avatar as *const _);
|
||||
|
||||
instance
|
||||
}
|
||||
|
||||
pub fn get_logic_client_home(&self) -> &'static mut LogicClientHome {
|
||||
let logic_client_home = LogicClientHome::new();
|
||||
let mut stream = ByteStream::from(&self.logic_client_home);
|
||||
logic_client_home.decode(&mut stream);
|
||||
stream.destruct();
|
||||
|
||||
logic_client_home
|
||||
}
|
||||
|
||||
pub fn get_logic_client_avatar(&self) -> &'static mut LogicClientAvatar {
|
||||
let logic_client_avatar = LogicClientAvatar::new();
|
||||
let mut stream = ByteStream::from(&self.logic_client_avatar);
|
||||
(logic_client_avatar.vtable.decode)(logic_client_avatar, &mut stream);
|
||||
stream.destruct();
|
||||
|
||||
logic_client_avatar
|
||||
}
|
||||
|
||||
pub fn get_spell_deck(&self) -> &'static mut LogicSpellDeck {
|
||||
let logic_client_home = self.get_logic_client_home();
|
||||
let cloned_deck = logic_client_home.spell_deck.clone();
|
||||
logic_client_home.destruct();
|
||||
allocator::free(logic_client_home);
|
||||
|
||||
cloned_deck
|
||||
}
|
||||
}
|
75
libserver/src/util/data_stream.rs
Normal file
75
libserver/src/util/data_stream.rs
Normal file
|
@ -0,0 +1,75 @@
|
|||
use std::mem::MaybeUninit;
|
||||
|
||||
use crate::ffi_util::import;
|
||||
|
||||
#[repr(C)]
|
||||
pub struct ChecksumEncoder {
|
||||
pub vtable: usize,
|
||||
pub is_checksum_enabled: bool,
|
||||
pub checksum: i32,
|
||||
pub checksum_snapshot: i32,
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
pub struct ByteStream {
|
||||
pub parent: ChecksumEncoder,
|
||||
pub offset: i32,
|
||||
pub length: i32,
|
||||
pub bit_idx: i32,
|
||||
pub data: *const u8,
|
||||
pub capacity: i32,
|
||||
}
|
||||
|
||||
impl ByteStream {
|
||||
pub fn new(initial_capacity: usize) -> Self {
|
||||
import!(byte_stream_ctor(ptr: *mut ByteStream, initial_capacity: i32) -> () = 0x17B098 + 1);
|
||||
unsafe {
|
||||
let mut instance = MaybeUninit::zeroed().assume_init();
|
||||
byte_stream_ctor(&mut instance, initial_capacity as i32);
|
||||
instance
|
||||
}
|
||||
}
|
||||
|
||||
pub fn write_byte(&mut self, value: u8) {
|
||||
import!(byte_stream_write_byte(ptr: *mut ByteStream, value: u8) -> () = 0x17B7EC + 1);
|
||||
byte_stream_write_byte(self, value);
|
||||
}
|
||||
|
||||
pub fn reset_offset(&mut self) {
|
||||
self.offset = 0;
|
||||
self.bit_idx = 0;
|
||||
}
|
||||
|
||||
pub fn get_length(&self) -> usize {
|
||||
std::cmp::max(self.offset, self.length) as usize
|
||||
}
|
||||
|
||||
pub fn get_data(&self) -> &[u8] {
|
||||
unsafe { std::slice::from_raw_parts(self.data, self.get_length()) }
|
||||
}
|
||||
|
||||
pub fn destruct(mut self) {
|
||||
import!(byte_stream_destruct(ptr: *mut ByteStream) -> () = 0x17BC20 + 1);
|
||||
byte_stream_destruct(&mut self);
|
||||
}
|
||||
}
|
||||
|
||||
impl<T> From<T> for ByteStream
|
||||
where
|
||||
T: AsRef<[u8]>,
|
||||
{
|
||||
fn from(value: T) -> Self {
|
||||
import!(byte_stream_ctor(ptr: *mut ByteStream, data: *const u8, length: i32) -> () = 0x17B0E0 + 1);
|
||||
|
||||
unsafe {
|
||||
let mut instance = MaybeUninit::zeroed().assume_init();
|
||||
byte_stream_ctor(
|
||||
&mut instance,
|
||||
value.as_ref().as_ptr(),
|
||||
value.as_ref().len() as i32,
|
||||
);
|
||||
|
||||
instance
|
||||
}
|
||||
}
|
||||
}
|
1
libserver/src/util/mod.rs
Normal file
1
libserver/src/util/mod.rs
Normal file
|
@ -0,0 +1 @@
|
|||
pub mod data_stream;
|
BIN
screenshot.png
Normal file
BIN
screenshot.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 789 KiB |
Loading…
Reference in a new issue