├── .gitignore
├── CMakeLists.txt
├── LICENSE
├── README.md
├── README_en.md
├── README_jp.md
├── TargetList.json
├── TargetList_jp.json
└── UnityFPSUnlocker
├── CMakeLists.txt
├── file_watch
├── dispatcher
│ ├── epoller.cc
│ ├── epoller.hh
│ ├── ibusiness_event.hh
│ └── ipoller.hh
├── listener.cc
└── listener.hh
├── fpslimiter.cc
├── fpslimiter.hh
├── jni_helper
├── jni_helper.cc
└── jni_helper.hh
├── main.cc
├── main.hh
├── third
├── rapidjson
│ └── rapidjson
│ │ ├── allocators.h
│ │ ├── cursorstreamwrapper.h
│ │ ├── document.h
│ │ ├── encodedstream.h
│ │ ├── encodings.h
│ │ ├── error
│ │ ├── en.h
│ │ └── error.h
│ │ ├── filereadstream.h
│ │ ├── filewritestream.h
│ │ ├── fwd.h
│ │ ├── internal
│ │ ├── biginteger.h
│ │ ├── clzll.h
│ │ ├── diyfp.h
│ │ ├── dtoa.h
│ │ ├── ieee754.h
│ │ ├── itoa.h
│ │ ├── meta.h
│ │ ├── pow10.h
│ │ ├── regex.h
│ │ ├── stack.h
│ │ ├── strfunc.h
│ │ ├── strtod.h
│ │ └── swap.h
│ │ ├── istreamwrapper.h
│ │ ├── memorybuffer.h
│ │ ├── memorystream.h
│ │ ├── msinttypes
│ │ ├── inttypes.h
│ │ └── stdint.h
│ │ ├── ostreamwrapper.h
│ │ ├── pointer.h
│ │ ├── prettywriter.h
│ │ ├── rapidjson.h
│ │ ├── reader.h
│ │ ├── schema.h
│ │ ├── stream.h
│ │ ├── stringbuffer.h
│ │ └── writer.h
├── riru_hide
│ ├── hide.cc
│ ├── hide.hh
│ ├── pmparser.c
│ ├── pmparser.h
│ └── riru_hide_logger.hh
├── xdl
│ ├── CMakeLists.txt
│ ├── xdl.c
│ ├── xdl.h
│ ├── xdl.map.txt
│ ├── xdl_iterate.c
│ ├── xdl_iterate.h
│ ├── xdl_linker.c
│ ├── xdl_linker.h
│ ├── xdl_lzma.c
│ ├── xdl_lzma.h
│ ├── xdl_util.c
│ └── xdl_util.h
└── zygisk.hh
├── unity
├── unity_engine.cc
└── unity_engine.hh
└── utility
├── config.cc
├── config.hh
├── houdini.cc
├── houdini.hh
├── logger.hh
├── native_bridge.h
├── singleton.hh
├── socket.cc
└── socket.hh
/.gitignore:
--------------------------------------------------------------------------------
1 | .vs*
2 | build
3 | libs
4 | riru_module
5 | zygisk_module
6 | *.cmd
7 | *.zip
8 | *.json
--------------------------------------------------------------------------------
/CMakeLists.txt:
--------------------------------------------------------------------------------
1 | # CMakeList.txt: 顶层 CMake 项目文件,在此处执行全局配置
2 | # 并包含子项目。
3 | #
4 | cmake_minimum_required (VERSION 3.21)
5 |
6 | project ("UnityFPSUnlocker")
7 |
8 | # 包含子项目。
9 | add_subdirectory ("UnityFPSUnlocker")
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2022 hexstr
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # UnityFPSUnlocker
2 | [README_EN](https://github.com/hexstr/UnityFPSUnlocker/blob/zygisk_module/README_en.md)
3 | [README_JP](https://github.com/hexstr/UnityFPSUnlocker/blob/zygisk_module/README_jp.md)
4 |
5 | ## 安装需求
6 | - 设备已安装 [Magisk](https://github.com/topjohnwu/Magisk/releases) / [KernelSU](https://github.com/tiann/KernelSU/releases) / [APatch](https://github.com/bmax121/APatch/releases)
7 | - 启用`Zygisk` (对于KernelSU / APatch用户 需要安装`Zygisk Next`)
8 | - `Xposed`版本需要`LSPosed`
9 |
10 | ## Xposed模块
11 | 一般`Xposed模块`不会随`Magisk模块`一起更新,所以请在[releases](https://github.com/hexstr/UnityFPSUnlocker/releases)中往前翻,寻找`app-debug.apk`。
12 | 你可以提取最新的`zygisk_module.zip/zygisk/{对应架构}.so`替换掉`app-debug.apk/lib/{对应架构}/libUnityFPSUnlocker.so`,然后用mt管理器重新签名。
13 |
14 | ## 附加需求
15 | - 对目标游戏启用`Shamiko` (可以避免一些游戏对于`Zygisk`注入或`SafetyNet`的检测)
16 |
17 | ## 使用
18 | 若你的手机没有开启`Zygisk`,请先开启`Zygisk`后重启。当上述需求准备完毕后即可刷入`UnityFPSUnlocker`模块。在重启之前,先下载`TargetList.json`放入`/data/local/tmp/TargetList.json`,并修改你的配置。
19 | 模块会通过判断游戏数据目录是否存在`/sdcard/Android/data/{包名}/files/il2cpp`来自动加载。
20 |
21 | ```
22 | {
23 | "global": {
24 | "delay": 10,
25 | "mod_opcode": true,
26 | "fps": 90,
27 | "scale": 1.0
28 | },
29 | "custom": {
30 | "com.random.package.name.a": {
31 | "fps": 60
32 | },
33 | "com.random.package.name.b": {
34 | "mod_opcode": false
35 | },
36 | "com.random.package.name.c": {
37 | "delay": 5
38 | }
39 | }
40 | }
41 | ```
42 |
43 | 其中,`global`节点中的配置为:
44 |
45 | - `fps` 需要设置的`fps`,设置为`0`以禁用
46 | - `delay` 游戏载入后等待`delay`秒执行
47 | - `mod_opcode` 是否修改`opcode`,如果你发现游戏会重新锁定fps,可以把这项改为`true`,但由于修改内存,可能会被反作弊检测到
48 | - `scale` 设置分辨率的倍数,一般保持`1.0`即可,必须为小数。`当前屏幕宽度 * scale x 当前屏幕高度 * scale`
49 |
50 | 然后,`custom`节点中的配置会覆盖`global`中的配置单独生效:
51 |
52 | - `key` 包名,比如`com.random.package.name.a`
53 | - `fps` 同上
54 | - `mod_opcode` 同上
55 | - `delay` 同上
56 | - `scale` 同上
57 |
58 | `TargetList.json`修改后可以搜索`json 格式校验`校验是否完整。修改后立即生效(`模块版本>=1.8`)。
59 | 可以在终端模拟器输入`logcat -s UnityFPSUnlocker`查看输入日志。
60 |
61 | ## 黑名单
62 | - 对于`Magisk模块`,设置对应的`fps`为`0`即可,插件将不会生效
63 | - 对于`Xposed模块`,在`LSPosed`中取消勾选对应的应用即可,如果没有显示应用,可以点击右上角选择`隐藏->游戏`
64 |
65 | ## 分辨率
66 | 最近发现`BlueArchive`最高分辨率只有`1080P`,在模拟器上有肉眼可见的锯齿,而且在`16:10`的比例下甚至像素点比`16:9`更少
67 |
68 | `2560x1600->1822x1138`
69 |
70 | 对比
71 |
72 | `2560x1440->1920x1080`
73 |
74 | 所以加上调整分辨率的功能,对比图如下
75 |
76 | https://imgsli.com/MjI3NDQ2/0/1
77 |
78 | https://imgsli.com/MjI3NDQ2/2/3
79 |
80 | 但是设置的时机需要尽可能早,也就是`delay`尽可能短,否则需要更改`绘图`选项中的任意一项后生效(比如开关一次`抗锯齿`)
81 |
82 | 如果你不需要超分辨率,可以修改`/sdcard/Android/data/com.nexon.bluearchive/files/DeviceOption`,把`Resolution`修改为大于`3`即可,这样会走`default`分支(在函数`GraphicsManager.CoSetScreenResolution()`中)
83 |
84 | - `0`: 1080P
85 | - `1`: 720P
86 | - `2`: 540P
87 | - `3`: 480P
88 | - `default` default
89 |
--------------------------------------------------------------------------------
/README_en.md:
--------------------------------------------------------------------------------
1 | # UnityFPSUnlocker
2 | ## Installation requirements
3 | - The device has already installed [Magisk](https://github.com/topjohnwu/Magisk/releases) / [KernelSU](https://github.com/tiann/KernelSU/releases) / [APatch](https://github.com/bmax121/APatch/releases)
4 | - Enable `Zygisk` (For KernelSU / APatch user, Need install `Zygisk Next`)
5 |
6 | ## Additional requirements
7 | - Enable `Shamiko` for the target game (Shamiko can avoid detected Zygisk injection or SafetyNet in some games)
8 |
9 | ## Use now
10 | If your device does not have Zygisk enabled, please enable Zygisk first and then reboot your device. After the above requirements are prepared, the 'UnityFPSUnlocker' module can be install in. Before rebooting, download `TargetList.json` and place it to `/data/local/tmp/TargetList.json`, and modify configuration by yourself.
11 | The module will automatically load by checking if the game data directory contains `/sdcard/Android/data/{package_name}/files/il2cpp`.
12 |
13 | ```
14 | {
15 | "global": {
16 | "delay": 10,
17 | "mod_opcode": true,
18 | "fps": 90,
19 | "scale": 1.0
20 | },
21 | "custom": {
22 | "com.random.package.name.a": {
23 | "fps": 60
24 | },
25 | "com.random.package.name.b": {
26 | "mod_opcode": false
27 | },
28 | "com.random.package.name.c": {
29 | "delay": 5
30 | }
31 | }
32 | }
33 | ```
34 |
35 | Among them, the configuration in the `global` node is:
36 |
37 | - `fps` Required settings for `fps`,Set to `0` to disable module injection
38 | - `delay` Wait for `delay` seconds to execute after the game is loaded
39 | - `mod_opcode` Do you want to modify `opcode`, If you find that the game will lock fps again, you can change this to `true`, But due to modifying memory, it maybe detected by anti-cheat
40 | - `scale` Set the multiplier of resolution, Generally, it keep `1.0`, and which must be a decimal number. `Current screen width * scale x Current screen height * scale`
41 |
42 | Then, the configuration in the `custom` node will override the configuration in the `global` and take effect separately:
43 |
44 | - `key` package name, like `com.random.package.name.a`
45 | - `fps` Same as above
46 | - `mod_opcode` Same as above
47 | - `delay` Same as above
48 | - `scale` Same as above
49 |
50 | `TargetList.json` effective immediately after modification (`Module version>=1.8`)。
51 | And you can input `logcat - s UnityFPSUnlocker` in the shell to view the input logs.
52 |
--------------------------------------------------------------------------------
/README_jp.md:
--------------------------------------------------------------------------------
1 | # UnityFPSUnlocker
2 | ## インストール条件
3 | - [Magisk](https://github.com/topjohnwu/Magisk/releases) / [KernelSU](https://github.com/tiann/KernelSU/releases) / [APatch](https://github.com/bmax121/APatch/releases)
4 | - `Zygisk`の有効化 (KernelSU / APatch ユーザーには `Zygisk Next`)
5 | - `Shamiko`の有効化
6 |
ShamikoはSafetyNetによる対策が入ったゲームに適用する場合に必要になる事があります。(ウマ娘など)
7 |
8 | ## 使用方法
9 | UnityFPSUnlockerとZygiskの有効化を行ない、`TargetList.json`を`/data/local/tmp/`に配置をしてください。
10 | プラグインは`/sdcard/Android/data/{パッケージ名}/files/il2cpp`にファイルが入っていると自動的にロードされます。
11 |
12 | ```
13 | {
14 | "global": {
15 | "delay": 10,
16 | "mod_opcode": true,
17 | "fps": 90
18 | },
19 | "custom": {
20 | "com.random.package.name.a": {
21 | "fps": 60
22 | },
23 | "com.random.package.name.b": {
24 | "mod_opcode": false
25 | },
26 | "com.random.package.name.c": {
27 | "delay": 5
28 | }
29 | }
30 | }
31 | ```
32 |
33 | 以下は`global`ノードの設定方法です:
34 | - `fps`はアンロックを行なう`FPS値`を入力します。
35 | - `delay`はアンロックを行なうまでの`ディレイ値`の設定です。
36 | - `mod_opcode`は`オペコード`の変更モードです。
37 | FPSを再ロックしてしまうゲームでこのオプションを`ture`にする事で、完全なアンロックが可能になりますがメモリの変更を行なうため「アンチチート機能」が反応してしまう事があります。
38 |
39 | `custom`ノードの設定は`global`ノードの設定を上書きし、別々で有効化されます:
40 | - `キー`となるパッケージ名、例:`com.random.package.name.a`
41 | - `fps` `global`ノード設定方法と同上
42 | - `mod_opcode` `global`ノード設定方法と同上
43 | - `delay` `global`ノード設定方法と同上
44 |
45 | `TargetList.json`に`JSONのフォーマットのチェック`を行なうようにしました。(>=1.8)
46 | 動作のログはターミナルエミュレーターで`logcat -s UnityFPSUnlocker`を入力する事でログを確認できます。
47 |
--------------------------------------------------------------------------------
/TargetList.json:
--------------------------------------------------------------------------------
1 | {
2 | "global": {
3 | "delay": 10,
4 | "mod_opcode": true,
5 | "fps": 120
6 | },
7 | "custom": {
8 | "com.nexon.bluearchive": {
9 | "fps": 90
10 | },
11 | "com.random.package.name.b": {
12 | "mod_opcode": false
13 | },
14 | "com.random.package.name.c": {
15 | "delay": 5
16 | },
17 | "com.random.package.name.d": {
18 | "description": "keep blank to use global parameters (delete this line)."
19 | }
20 | }
21 | }
22 |
--------------------------------------------------------------------------------
/TargetList_jp.json:
--------------------------------------------------------------------------------
1 | {
2 | "global": {
3 | "delay": 10,
4 | "mod_opcode": true,
5 | "fps": 120
6 | },
7 | "custom": {
8 | "com.nexon.bluearchive": {
9 | "fps": 120
10 | },
11 | "com.YoStarJP.AzurLane": {
12 | "mod_opcode": false
13 | },
14 | "com.YostarJP.BlueArchive": {
15 | "mod_opcode": false
16 | },
17 | "com.sunborn.girlsfrontline.jp": {
18 | "mod_opcode": false
19 | },
20 | "com.miHoYo.GenshinImpact": {
21 | "mod_opcode": false
22 | },
23 | "co.grimoire.bxb": {
24 | "mod_opcode": false
25 | },
26 | "jp.co.craftegg.band": {
27 | "mod_opcode": false
28 | },
29 | "com.proximabeta.nikke": {
30 | "mod_opcode": false
31 | },
32 | "com.random.package.name.b": {
33 | "mod_opcode": false
34 | },
35 | "com.random.package.name.c": {
36 | "fps": 120
37 | },
38 | "com.random.package.name.d": {
39 | "記述": "グローバルパラメータを使用する場合は空白にします(この行は削除してください)"
40 | }
41 | }
42 | }
--------------------------------------------------------------------------------
/UnityFPSUnlocker/CMakeLists.txt:
--------------------------------------------------------------------------------
1 | # CMakeList.txt: UnityFPSUnlocker 的 CMake 项目,在此处包括源代码并定义
2 | # 项目特定的逻辑。
3 | #
4 | cmake_minimum_required (VERSION 3.21)
5 |
6 | set(CMAKE_CXX_STANDARD 17)
7 | set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fno-rtti -fno-exceptions -D_LIBCPP_DISABLE_VISIBILITY_ANNOTATIONS")
8 |
9 | set(LibraryName UnityFPSUnlocker)
10 |
11 | # 隐藏导出函数
12 | set(CMAKE_C_VISIBILITY_PRESET hidden)
13 | set(CMAKE_CXX_VISIBILITY_PRESET hidden)
14 |
15 | # 添加Debug和Release版本信息
16 | string(TOUPPER ${CMAKE_BUILD_TYPE} CMAKE_BUILD_TYPE_UP)
17 | if(CMAKE_BUILD_TYPE_UP STREQUAL DEBUG)
18 | message(STATUS "Debug Version")
19 | set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -g")
20 | set(CMAKE_SHARED_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -g")
21 | else()
22 | message(STATUS "Release Version")
23 | set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Oz -flto -fdata-sections -ffunction-sections")
24 | set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Oz -flto -fdata-sections -ffunction-sections")
25 | set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -s -Wl,--exclude-libs,ALL -flto")
26 | set(CMAKE_SHARED_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -s -Wl,--exclude-libs,ALL -Wl,--gc-sections -flto")
27 | endif()
28 |
29 | set(CMAKE_PREFIX_PATH "${CMAKE_SYSROOT}/${CMAKE_LIBRARY_ARCHITECTURE}")
30 | message(STATUS CMAKE_PREFIX_PATH: ${CMAKE_PREFIX_PATH})
31 |
32 | # 添加头文件
33 | include_directories(
34 | .
35 | "third/rapidjson"
36 | "third/xdl"
37 | )
38 |
39 | # 文件写出位置
40 | set(LIBRARY_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/libs/${CMAKE_BUILD_TYPE}/)
41 |
42 | # 将源代码添加到此项目的可执行文件。
43 | aux_source_directory(${PROJECT_SOURCE_DIR}/${LibraryName} program-src)
44 | aux_source_directory(unity unity-src)
45 | aux_source_directory(utility utility-src)
46 | aux_source_directory(third/xdl xdl-src)
47 | aux_source_directory(third/riru_hide riru_hide-src)
48 | aux_source_directory(file_watch listener-src)
49 | aux_source_directory(file_watch/dispatcher dispatcher-src)
50 |
51 | add_library (
52 | ${LibraryName}
53 | SHARED
54 | ${program-src}
55 | ${unity-src}
56 | ${utility-src}
57 | ${listener-src}
58 | ${dispatcher-src}
59 | ${xdl-src}
60 | ${riru_hide-src}
61 | )
62 |
63 | find_library(log-lib log)
64 | target_link_libraries(${LibraryName} ${log-lib})
65 |
66 | find_package(absl REQUIRED)
67 | target_link_libraries(${LibraryName} absl::status)
68 | target_link_libraries(${LibraryName} absl::statusor)
69 | target_link_libraries(${LibraryName} absl::flat_hash_set)
70 |
71 | set_target_properties(${LibraryName} PROPERTIES PREFIX "")
72 | set_target_properties(${LibraryName} PROPERTIES OUTPUT_NAME ${ANDROID_ABI})
--------------------------------------------------------------------------------
/UnityFPSUnlocker/file_watch/dispatcher/epoller.cc:
--------------------------------------------------------------------------------
1 | #ifdef __linux__
2 |
3 | #include
4 |
5 | #include
6 |
7 | #include "epoller.hh"
8 | #include "utility/logger.hh"
9 |
10 | std::vector EPoller::reserved_list_;
11 |
12 | EPoller::EPoller(IBusinessEvent* business) {
13 | epoller_inst_ = epoll_create1(0);
14 | op_ = business;
15 | op_->poller_ = reinterpret_cast(this);
16 | }
17 |
18 | int EPoller::SetNonBlocking(int fd) {
19 | int old_opt = fcntl(fd, F_GETFL);
20 | int new_opt = old_opt | O_NONBLOCK;
21 | return fcntl(fd, F_SETFL, new_opt);
22 | }
23 |
24 | absl::Status EPoller::AddSocket(intptr_t s, long eventflags) {
25 | epoll_event _event;
26 | _event.data.u64 = s;
27 | _event.events = eventflags;
28 | if (epoll_ctl(epoller_inst_, EPOLL_CTL_ADD, s, &_event) == -1) {
29 | return absl::InternalError(strerror(errno));
30 | }
31 | return absl::OkStatus();
32 | }
33 |
34 | absl::Status EPoller::ModSocket(intptr_t s, long eventflags) {
35 | epoll_event _event;
36 | _event.data.fd = s;
37 | _event.events = eventflags;
38 | if (epoll_ctl(epoller_inst_, EPOLL_CTL_MOD, s, &_event) == -1) {
39 | return absl::InternalError(strerror(errno));
40 | }
41 | return absl::OkStatus();
42 | }
43 |
44 | absl::Status EPoller::RemoveSocket(intptr_t s) {
45 | if (epoll_ctl(epoller_inst_, EPOLL_CTL_DEL, s, nullptr) == -1) {
46 | return absl::InternalError(strerror(errno));
47 | }
48 | else {
49 | return absl::OkStatus();
50 | }
51 | }
52 |
53 | void EPoller::Poll() {
54 | int CompEventNum = epoll_wait(epoller_inst_, &event_array_[0], MAX_EVENT_NUMBER, 3000);
55 | for (int i = 0; i < CompEventNum; ++i) {
56 | HandleEvents(reinterpret_cast(event_array_[i].data.ptr), event_array_[i].events);
57 | }
58 | }
59 |
60 | void EPoller::HandleEvents(intptr_t s, uint32_t event) {
61 | if (event & EPOLLRDHUP) {
62 | op_->OnClose(s);
63 | }
64 |
65 | else if (event & EPOLLIN) {
66 | op_->OnReadable(s);
67 | }
68 |
69 | else if (event & EPOLLOUT) {
70 | op_->OnWritable(s);
71 | }
72 |
73 | else if (event & EPOLLERR) {
74 | op_->OnError(s);
75 | }
76 | }
77 |
78 | #endif // #ifdef __linux__
--------------------------------------------------------------------------------
/UnityFPSUnlocker/file_watch/dispatcher/epoller.hh:
--------------------------------------------------------------------------------
1 | #ifndef EPOLLER_HEADER
2 | #define EPOLLER_HEADER
3 |
4 | #include
5 | #ifdef __linux__
6 |
7 | #include
8 |
9 | #include
10 | #include
11 |
12 | #include "ipoller.hh"
13 |
14 | constexpr int MAX_EVENT_NUMBER = 64;
15 |
16 | class EPoller : public IPoller {
17 | public:
18 | EPoller(IBusinessEvent* business);
19 | ~EPoller(){};
20 |
21 | absl::Status AddSocket(intptr_t s, long eventflags) override;
22 | absl::Status ModSocket(intptr_t s, long eventflags) override;
23 | absl::Status RemoveSocket(intptr_t s) override;
24 |
25 | void Poll() override;
26 | void HandleEvents(intptr_t s, uint32_t event);
27 |
28 | std::array event_array_{};
29 |
30 | static int SetNonBlocking(int);
31 | static std::vector reserved_list_; // SubReactor list
32 |
33 | protected:
34 | int epoller_inst_ = 0;
35 | };
36 |
37 | #endif // #ifdef __linux__
38 |
39 | #endif // epoller.hh
--------------------------------------------------------------------------------
/UnityFPSUnlocker/file_watch/dispatcher/ibusiness_event.hh:
--------------------------------------------------------------------------------
1 | #ifndef IBUSINESS_EVENT_HEADER
2 | #define IBUSINESS_EVENT_HEADER
3 |
4 | #include
5 |
6 | class IPoller;
7 |
8 | class IBusinessEvent {
9 | public:
10 | virtual ~IBusinessEvent() {}
11 | virtual void OnAcceptable(uintptr_t){};
12 | virtual void OnClose(uintptr_t){};
13 | virtual void OnReadable(uintptr_t){};
14 | virtual void OnWritable(uintptr_t){};
15 | virtual void OnError(uintptr_t){};
16 | IPoller* poller_ = nullptr;
17 | };
18 |
19 | #endif // ibusiness_event.hh
--------------------------------------------------------------------------------
/UnityFPSUnlocker/file_watch/dispatcher/ipoller.hh:
--------------------------------------------------------------------------------
1 | #ifndef IPOLLER_HEADER
2 | #define IPOLLER_HEADER
3 |
4 | #include
5 |
6 | #include "ibusiness_event.hh"
7 |
8 | class IPoller {
9 | public:
10 | IPoller(){};
11 | virtual ~IPoller(){};
12 | virtual absl::Status AddSocket(intptr_t s, long eventflags) = 0;
13 | virtual absl::Status ModSocket(intptr_t s, long eventflags) = 0;
14 | virtual absl::Status RemoveSocket(intptr_t s) = 0;
15 | // while (IsRunning) {
16 | virtual void Poll() = 0;
17 | // }
18 | void SetBusiness(IBusinessEvent* op) { op_ = op; }
19 |
20 | protected:
21 | IBusinessEvent* op_ = nullptr;
22 | };
23 |
24 | #endif // ipoller.hh
--------------------------------------------------------------------------------
/UnityFPSUnlocker/file_watch/listener.cc:
--------------------------------------------------------------------------------
1 | #include "listener.hh"
2 |
3 | #include
4 | #include
5 | #include
6 | #include
7 | #include
8 | #include
9 |
10 | #include "dispatcher/epoller.hh"
11 | #include "utility/logger.hh"
12 |
13 | namespace FileWatch {
14 | Listener::Listener() {
15 | inotify_fd_ = inotify_init1(IN_NONBLOCK | IN_CLOEXEC);
16 | if (inotify_fd_ == -1) {
17 | ERROR("[Listener::Register] [%d] %s", __LINE__, strerror(errno));
18 | }
19 | }
20 |
21 | int Listener::Register(const char* file_path, OnModified mod_callback, OnDeleted del_callback) {
22 | int wd = inotify_add_watch(inotify_fd_, file_path, IN_CLOSE_WRITE | IN_MODIFY | IN_DELETE_SELF);
23 | if (wd == -1) {
24 | ERROR("[Listener::Register] [%d] %s", __LINE__, strerror(errno));
25 | return -1;
26 | }
27 |
28 | auto res = EPoller::reserved_list_[0]->AddSocket(inotify_fd_, EPOLLIN);
29 | if (!res.ok()) {
30 | ERROR("[Listener::Register] %s", res.message().data());
31 | return -1;
32 | }
33 |
34 | registered_[wd] = std::make_tuple(mod_callback, del_callback);
35 | LOG("[Listener::Register] Registered: wd: %d %s", wd, file_path);
36 | return wd;
37 | }
38 |
39 | void Listener::Remove(int wd) {
40 | inotify_rm_watch(inotify_fd_, wd);
41 | }
42 |
43 | void Listener::OnReadable(uintptr_t s) {
44 | inotify_event event;
45 | int read_size = read(s, &event, sizeof(event));
46 | if (read_size == -1) {
47 | ERROR("[Listener::OnReadable] [%d] %s", __LINE__, strerror(errno));
48 | return;
49 | }
50 | /*
51 | if (event.mask & IN_CREATE) {
52 | LOGI("The file was created.");
53 | }
54 | else*/
55 | if (event.mask & IN_DELETE_SELF) {
56 | if (auto itor = registered_.find(event.wd); itor != registered_.end()) {
57 | auto del_callback = std::get<1>(itor->second);
58 | if (del_callback != nullptr) {
59 | del_callback();
60 | }
61 | }
62 | }
63 | /*
64 | else if (event.mask & IN_MODIFY) {
65 | LOGI("The file was modified.");
66 | }
67 | */
68 | else if (event.mask & IN_CLOSE_WRITE) {
69 | if (auto itor = registered_.find(event.wd); itor != registered_.end()) {
70 | auto mod_callback = std::get<0>(itor->second);
71 | if (mod_callback != nullptr) {
72 | mod_callback(event.wd);
73 | }
74 | }
75 | }
76 | }
77 | } // namespace FileWatch
--------------------------------------------------------------------------------
/UnityFPSUnlocker/file_watch/listener.hh:
--------------------------------------------------------------------------------
1 | #ifndef LISTENER_HEADER
2 | #define LISTENER_HEADER
3 |
4 | #include
5 |
6 | #include "dispatcher/ibusiness_event.hh"
7 |
8 | #include
9 |
10 | namespace FileWatch {
11 | class Listener : public IBusinessEvent {
12 | public:
13 | Listener();
14 | virtual void OnReadable(uintptr_t) override final;
15 | virtual void OnWritable(uintptr_t) override final{};
16 | virtual void OnError(uintptr_t) override final{};
17 | virtual void OnClose(uintptr_t) override final{};
18 |
19 | using OnModified = void (*)(int wd);
20 | using OnDeleted = void (*)();
21 | int Register(const char* file_path, OnModified mod_callback, OnDeleted del_callback);
22 | void Remove(int wd);
23 |
24 | private:
25 | int inotify_fd_;
26 | absl::flat_hash_map> registered_;
27 | };
28 | } // namespace FileWatch
29 |
30 | #endif // listener.hh
--------------------------------------------------------------------------------
/UnityFPSUnlocker/fpslimiter.cc:
--------------------------------------------------------------------------------
1 | #include "fpslimiter.hh"
2 |
3 | #include
4 | #include
5 | #include
6 |
7 | #include "unity/unity_engine.hh"
8 | #include "utility/logger.hh"
9 |
10 | #include
11 |
12 | namespace FPSLimiter {
13 | void Start(const ConfigValue& cfg) {
14 | #ifdef __aarch64__
15 | LOG("[UnityFPSUnlocker][arm64] Starting...");
16 | #elif defined(__ARM_ARCH_7A__)
17 | LOG("[UnityFPSUnlocker][armv7] Starting...");
18 | #elif defined(__i386__)
19 | LOG("[UnityFPSUnlocker][x86] Starting...");
20 | #elif defined(__x86_64__)
21 | LOG("[UnityFPSUnlocker][x86_64] Starting...");
22 | #endif
23 | cfg.DebugPrint();
24 | std::chrono::seconds sleep_duration(cfg.delay_);
25 | std::this_thread::sleep_for(sleep_duration);
26 |
27 | LOG("***** begin *****");
28 | void* handle = nullptr;
29 | for (int i = 0; i < 10; ++i) {
30 | if ((handle = xdl_open("libil2cpp.so", 0))) {
31 | break;
32 | }
33 | std::this_thread::sleep_for(std::chrono::seconds(1));
34 | }
35 | if (handle) {
36 | Unity::GetInstance().Init(handle);
37 | Unity::GetInstance().SetFrameRate(cfg.fps_, cfg.mod_opcode_);
38 | Unity::GetInstance().SetResolution(cfg.scale_);
39 |
40 | LOG("***** finished *****");
41 | return;
42 | }
43 | else {
44 | ERROR("Failed to open libil2cpp.so");
45 | }
46 | }
47 | } // namespace FPSLimiter
48 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/fpslimiter.hh:
--------------------------------------------------------------------------------
1 | #ifndef FPSLIMITER_HEADER
2 | #define FPSLIMITER_HEADER
3 |
4 | #include "utility/config.hh"
5 |
6 | namespace FPSLimiter {
7 | void Start(const ConfigValue&);
8 | } // namespace FPSLimiter
9 |
10 | #endif // fpslimiter.hh
--------------------------------------------------------------------------------
/UnityFPSUnlocker/jni_helper/jni_helper.cc:
--------------------------------------------------------------------------------
1 | #include "jni_helper.hh"
2 |
3 | namespace JNIHelper {
4 | jobject GetDisplay(JNIEnv* env, jobject activity) {
5 | jclass activity_class = env->GetObjectClass(activity);
6 | JNIEX(env, activity_class);
7 |
8 | jclass build_version_class = env->FindClass("android/os/Build$VERSION");
9 | JNIEX(env, build_version_class);
10 |
11 | jfieldID sdk_int_field = env->GetStaticFieldID(build_version_class, "SDK_INT", "I");
12 | jint sdk_int = env->GetStaticIntField(build_version_class, sdk_int_field);
13 |
14 | if (sdk_int >= 30) {
15 | jmethodID get_display_method = env->GetMethodID(activity_class, "getDisplay", "()Landroid/view/Display;");
16 | JNIEX(env, get_display_method);
17 | return env->CallObjectMethod(activity, get_display_method);
18 | }
19 | else {
20 | jclass context_class = env->FindClass("android/content/Context");
21 | JNIEX(env, context_class);
22 | jfieldID window_service_field = env->GetStaticFieldID(context_class, "WINDOW_SERVICE", "Ljava/lang/String;");
23 | jobject window_service_string = env->GetStaticObjectField(context_class, window_service_field);
24 |
25 | jmethodID get_system_service = env->GetMethodID(activity_class, "getSystemService", "(Ljava/lang/String;)Ljava/lang/Object;");
26 | JNIEX(env, get_system_service);
27 | jobject window_manager = env->CallObjectMethod(activity, get_system_service);
28 |
29 | jclass window_manager_class = env->GetObjectClass(window_manager);
30 | JNIEX(env, window_manager_class);
31 | jmethodID get_default_display_method = env->GetMethodID(window_manager_class, "getDefaultDisplay", "()Landroid/view/Display;");
32 | JNIEX(env, get_default_display_method);
33 | return env->CallObjectMethod(window_manager, get_default_display_method);
34 | }
35 | }
36 |
37 | jobject GetActivity(JNIEnv* env) {
38 | jclass unity_player_class = env->FindClass("com/unity3d/player/UnityPlayer");
39 | JNIEX(env, unity_player_class);
40 |
41 | jfieldID current_activity_field = env->GetStaticFieldID(unity_player_class, "currentActivity", "Landroid/app/Activity;");
42 |
43 | jobject current_activity = env->GetStaticObjectField(unity_player_class, current_activity_field);
44 | JNIEX(env, current_activity);
45 |
46 | return current_activity;
47 | }
48 |
49 | jint GetPreferredModeId(JNIEnv* env, jobject activity) {
50 | jclass activity_class = env->GetObjectClass(activity);
51 | jmethodID get_window_method = env->GetMethodID(activity_class, "getWindow", "()Landroid/view/Window;");
52 | jobject window = env->CallObjectMethod(activity, get_window_method);
53 |
54 | jclass window_class = env->GetObjectClass(window);
55 | jmethodID get_attributes_method = env->GetMethodID(window_class, "getAttributes", "()Landroid/view/WindowManager$LayoutParams;");
56 | jobject params = env->CallObjectMethod(window, get_attributes_method);
57 |
58 | jclass layout_params_class = env->GetObjectClass(params);
59 | jfieldID preferred_mode_id_field = env->GetFieldID(layout_params_class, "preferredDisplayModeId", "I");
60 | jint preferred_mode_id = env->GetIntField(params, preferred_mode_id_field);
61 |
62 | return preferred_mode_id;
63 | }
64 |
65 | std::vector GetSupportedModes(JNIEnv* env, jobject activity) {
66 | std::vector ret;
67 |
68 | if (jobject display = GetDisplay(env, activity)) {
69 | jclass display_class = env->GetObjectClass(display);
70 | jmethodID get_supported_modes_method = env->GetMethodID(display_class, "getSupportedModes", "()[Landroid/view/Display$Mode;");
71 | jobjectArray modes_array = (jobjectArray)env->CallObjectMethod(display, get_supported_modes_method);
72 |
73 | jsize length = env->GetArrayLength(modes_array);
74 |
75 | jclass mode_class = env->FindClass("android/view/Display$Mode");
76 | jmethodID get_mode_id_method = env->GetMethodID(mode_class, "getModeId", "()I");
77 | jmethodID get_physical_width_method = env->GetMethodID(mode_class, "getPhysicalWidth", "()I");
78 | jmethodID get_physical_height_method = env->GetMethodID(mode_class, "getPhysicalHeight", "()I");
79 | jmethodID get_refresh_rate_method = env->GetMethodID(mode_class, "getRefreshRate", "()F");
80 |
81 | for (jsize i = 0; i < length; i++) {
82 | jobject mode_obj = env->GetObjectArrayElement(modes_array, i);
83 | Mode mode;
84 | mode.id = env->CallIntMethod(mode_obj, get_mode_id_method);
85 | mode.width = env->CallIntMethod(mode_obj, get_physical_width_method);
86 | mode.height = env->CallIntMethod(mode_obj, get_physical_height_method);
87 | mode.refresh_rate = env->CallFloatMethod(mode_obj, get_refresh_rate_method);
88 | ret.emplace_back(mode);
89 | env->DeleteLocalRef(mode_obj);
90 | }
91 | }
92 |
93 | return ret;
94 | }
95 |
96 | void SetPreferredMode(JNIEnv* env, jobject activity, jint mode) {
97 | jclass activity_class = env->GetObjectClass(activity);
98 | jmethodID get_window_method = env->GetMethodID(activity_class, "getWindow", "()Landroid/view/Window;");
99 | jobject window = env->CallObjectMethod(activity, get_window_method);
100 |
101 | jclass window_class = env->GetObjectClass(window);
102 | jmethodID get_attributes_method = env->GetMethodID(window_class, "getAttributes", "()Landroid/view/WindowManager$LayoutParams;");
103 | jobject params = env->CallObjectMethod(window, get_attributes_method);
104 |
105 | jclass layout_params_class = env->GetObjectClass(params);
106 | jfieldID preferred_mode_id_field = env->GetFieldID(layout_params_class, "preferredDisplayModeId", "I");
107 | env->SetIntField(params, preferred_mode_id_field, mode);
108 |
109 | jmethodID set_attributes_method = env->GetMethodID(window_class, "setAttributes", "(Landroid/view/WindowManager$LayoutParams;)V");
110 | env->CallVoidMethod(window, set_attributes_method, params);
111 | }
112 |
113 | void JNIExceptionHandler(JNIEnv* env) {
114 | jthrowable exc = env->ExceptionOccurred();
115 | if (exc) {
116 | env->ExceptionDescribe();
117 | env->ExceptionClear();
118 | }
119 | }
120 |
121 | } // namespace JNIHelper
--------------------------------------------------------------------------------
/UnityFPSUnlocker/jni_helper/jni_helper.hh:
--------------------------------------------------------------------------------
1 | #pragma once
2 |
3 | #include
4 |
5 | #include
6 |
7 | #include "utility/logger.hh"
8 |
9 | #define JNIEX(env, name) \
10 | if (name == nullptr) { \
11 | ERROR("nullptr " #name); \
12 | JNIExceptionHandler(env); \
13 | return nullptr; \
14 | }
15 |
16 | namespace JNIHelper {
17 | struct Mode {
18 | int id;
19 | int width;
20 | int height;
21 | float refresh_rate;
22 | };
23 |
24 | jobject GetDisplay(JNIEnv* env, jobject activity);
25 | jobject GetActivity(JNIEnv* env);
26 | jint GetPreferredModeId(JNIEnv* env, jobject activity);
27 | std::vector GetSupportedModes(JNIEnv* env, jobject activity);
28 | void SetPreferredMode(JNIEnv* env, jobject activity, jint mode);
29 |
30 | void JNIExceptionHandler(JNIEnv* env);
31 | } // namespace JNIHelper
--------------------------------------------------------------------------------
/UnityFPSUnlocker/main.cc:
--------------------------------------------------------------------------------
1 | #include "main.hh"
2 |
3 | #include
4 | #include
5 | #include
6 |
7 | #include
8 | #include
9 |
10 | #include
11 | #include
12 | #include
13 | #include
14 | #include
15 |
16 | #include "file_watch/dispatcher/epoller.hh"
17 | #include "file_watch/listener.hh"
18 | #include "fpslimiter.hh"
19 | #include "third/riru_hide/hide.hh"
20 | #include "utility/houdini.hh"
21 | #include "utility/socket.hh"
22 |
23 | using namespace rapidjson;
24 |
25 | static bool is_loaded = false;
26 | static int watch_descriptor = -1;
27 | static absl::flat_hash_map custom_list;
28 | static ConfigValue global_cfg;
29 | static FileWatch::Listener* file_watch_listener = nullptr;
30 |
31 | constexpr const char* ConfigFile = "/data/local/tmp/TargetList.json";
32 |
33 | absl::Status LoadConfig() {
34 | custom_list.clear();
35 |
36 | auto read_path = Utility::LoadJsonFromFile(ConfigFile);
37 | if (!read_path.ok()) {
38 | return read_path.status();
39 | }
40 |
41 | Document& doc = *read_path;
42 | if (auto itor = doc.FindMember("global"); itor != doc.MemberEnd() && itor->value.IsObject()) {
43 | const auto& itor_json_object = itor->value.GetObject();
44 | if (auto itor2 = itor_json_object.FindMember("delay"); itor2 != itor_json_object.MemberEnd() && itor2->value.IsInt()) {
45 | global_cfg.delay_ = itor2->value.GetInt();
46 | }
47 | if (auto itor2 = itor_json_object.FindMember("fps"); itor2 != itor_json_object.MemberEnd() && itor2->value.IsInt()) {
48 | global_cfg.fps_ = itor2->value.GetInt();
49 | }
50 | if (auto itor2 = itor_json_object.FindMember("mod_opcode"); itor2 != itor_json_object.MemberEnd() && itor2->value.IsBool()) {
51 | global_cfg.mod_opcode_ = itor2->value.GetBool();
52 | }
53 | if (auto itor2 = itor_json_object.FindMember("scale"); itor2 != itor_json_object.MemberEnd() && itor2->value.IsFloat()) {
54 | global_cfg.scale_ = itor2->value.GetFloat();
55 | }
56 | }
57 |
58 | if (auto itor = doc.FindMember("custom"); itor != doc.MemberEnd() && itor->value.IsObject()) {
59 | for (const auto& item : itor->value.GetObject()) {
60 | if (item.value.IsObject()) {
61 | auto cfg(global_cfg);
62 | if (item.value.MemberCount()) {
63 | if (auto itor2 = item.value.FindMember("delay"); itor2 != item.value.MemberEnd() && itor2->value.IsInt()) {
64 | cfg.delay_ = itor2->value.GetInt();
65 | }
66 | if (auto itor2 = item.value.FindMember("fps"); itor2 != item.value.MemberEnd() && itor2->value.IsInt()) {
67 | cfg.fps_ = itor2->value.GetInt();
68 | }
69 | if (auto itor2 = item.value.FindMember("mod_opcode"); itor2 != item.value.MemberEnd() && itor2->value.IsBool()) {
70 | cfg.mod_opcode_ = itor2->value.GetBool();
71 | }
72 | if (auto itor2 = item.value.FindMember("scale"); itor2 != item.value.MemberEnd() && itor2->value.IsFloat()) {
73 | cfg.scale_ = itor2->value.GetFloat();
74 | }
75 | }
76 | custom_list[item.name.GetString()] = cfg;
77 | }
78 | }
79 | }
80 |
81 | LOG("[LoadConfig] custom_list: %zu", custom_list.size());
82 | LOG("[LoadConfig] global_cfg: ");
83 | global_cfg.DebugPrint();
84 |
85 | return absl::OkStatus();
86 | }
87 |
88 | void OnModified(int wd) {
89 | if (wd == watch_descriptor) {
90 | LoadConfig().IgnoreError();
91 | }
92 | }
93 |
94 | void OnDeleted() {
95 | watch_descriptor = -1;
96 | }
97 |
98 | // In zygiskd memory.
99 | void CompanionEntry(int s) {
100 | std::string package_name = read_string(s);
101 | if (is_loaded == false) {
102 | if (auto res = LoadConfig(); res.ok()) {
103 | is_loaded = true;
104 | file_watch_listener = new FileWatch::Listener();
105 | EPoller* file_watch_poller = new EPoller(file_watch_listener);
106 | EPoller::reserved_list_.push_back(file_watch_poller);
107 | std::thread([=] {
108 | while (true) {
109 | file_watch_poller->Poll();
110 | }
111 | }).detach();
112 | watch_descriptor = file_watch_listener->Register(ConfigFile, OnModified, OnDeleted);
113 | }
114 | else {
115 | ERROR("LoadConfig error: %s", res.message().data());
116 | }
117 | }
118 |
119 | if (is_loaded && watch_descriptor == -1) {
120 | watch_descriptor = file_watch_listener->Register(ConfigFile, OnModified, OnDeleted);
121 | }
122 |
123 | if (auto itor = custom_list.find(package_name); itor != custom_list.end()) {
124 | write_int(s, 1);
125 | write_int(s, itor->second.delay_);
126 | write_int(s, itor->second.fps_);
127 | write_int(s, itor->second.mod_opcode_);
128 | write_float(s, itor->second.scale_);
129 | }
130 | else {
131 | write_int(s, 0);
132 | write_int(s, global_cfg.delay_);
133 | write_int(s, global_cfg.fps_);
134 | write_int(s, global_cfg.mod_opcode_);
135 | write_float(s, global_cfg.scale_);
136 | }
137 | }
138 |
139 | REGISTER_ZYGISK_MODULE(MyModule)
140 | REGISTER_ZYGISK_COMPANION(CompanionEntry)
141 |
142 | void MyModule::onLoad(Api* api, JNIEnv* env) {
143 | this->api = api;
144 | this->env = env;
145 | }
146 |
147 | void MyModule::preAppSpecialize(AppSpecializeArgs* args) {
148 | int client_socket = api->connectCompanion();
149 |
150 | package_name_ = env->GetStringUTFChars(args->nice_name, nullptr);
151 | write_string(client_socket, package_name_);
152 |
153 | has_custom_cfg_ = read_int(client_socket);
154 | current_cfg_.delay_ = read_int(client_socket);
155 | current_cfg_.fps_ = read_int(client_socket);
156 | current_cfg_.mod_opcode_ = read_int(client_socket);
157 | current_cfg_.scale_ = read_float(client_socket);
158 |
159 | close(client_socket);
160 | }
161 |
162 | void MyModule::ForHoudini() {
163 | #if defined(__i386__) || defined(__x86_64__)
164 | std::thread([=]() {
165 | std::chrono::seconds sleep_duration(current_cfg_.delay_);
166 | std::this_thread::sleep_for(sleep_duration);
167 | #ifdef __x86_64__
168 | #define libdir "/lib64/x86_64"
169 | #define library_name "arm64-v8a.so"
170 | #endif
171 |
172 | #ifdef __i386__
173 | #define libdir "/lib64/x86"
174 | #define library_name "armeabi-v7a.so"
175 | #endif
176 |
177 | auto vms = Utility::GetVM();
178 | if (!vms.ok()) {
179 | ERROR("%s", vms.status().message().data());
180 | return;
181 | }
182 |
183 | JNIEnv* env = nullptr;
184 | if (vms.value()->AttachCurrentThread(&env, nullptr) < 0) {
185 | ERROR("Cannot connect to JNI environment");
186 | return;
187 | }
188 |
189 | auto app_info = Utility::GetApplicationInfo(env);
190 | auto path = Utility::GetLibraryPath(env, app_info.value());
191 |
192 | if (!path.ok()) {
193 | ERROR("%s", vms.status().message().data());
194 | return;
195 | }
196 |
197 | if (path.value().find(libdir) == std::string::npos) {
198 | auto& houdini = Houdini::GetInstance();
199 | auto plugin = houdini.LoadLibrary("/data/local/tmp/gh@hexstr/UnityFPSUnlocker/" library_name, RTLD_NOW);
200 | if (plugin.ok()) {
201 | if (plugin.value() == nullptr) {
202 | ERROR("Failed to load library : %s", Houdini::GetInstance().GetError());
203 | return;
204 | }
205 | ConfigValue config(0, current_cfg_.fps_, current_cfg_.mod_opcode_, current_cfg_.scale_);
206 | if (auto result = houdini.CallJNI(plugin.value(), vms.value(), &config);
207 | !result.ok()) {
208 | ERROR("%s", plugin.status().message().data());
209 | }
210 | riru_hide("/data/local/tmp/gh@hexstr/UnityFPSUnlocker/" library_name);
211 | }
212 | else {
213 | ERROR("%s", plugin.status().message().data());
214 | }
215 | }
216 | else {
217 | FPSLimiter::Start(current_cfg_);
218 | }
219 | }).detach();
220 | #endif
221 | }
222 |
223 | void MyModule::postAppSpecialize(const AppSpecializeArgs* args) {
224 | auto path = absl::Substitute("/sdcard/Android/data/$0/files/il2cpp", package_name_);
225 | if ((has_custom_cfg_ || access(path.c_str(), F_OK) == 0) && current_cfg_.fps_ > 0) {
226 | #if defined(__ARM_ARCH_7A__) || defined(__aarch64__)
227 | std::thread([=]() {
228 | FPSLimiter::Start(current_cfg_);
229 | }).detach();
230 | #endif
231 | ForHoudini();
232 | }
233 | else {
234 | api->setOption(zygisk::Option::DLCLOSE_MODULE_LIBRARY);
235 | }
236 | env->ReleaseStringUTFChars(args->nice_name, package_name_);
237 | }
238 |
239 | #if defined(__ARM_ARCH_7A__) || defined(__aarch64__)
240 |
241 | extern "C" {
242 | JNIEXPORT void JNICALL Java_io_github_hexstr_UnityFPSUnlocker_MyModule_HelloWorld(JNIEnv* env, jobject obj, jint delay, jint fps, jint mod_opcode, jfloat scale) {
243 | LOG("[UnityFPSUnlocker][xposed] delay: %d | fps: %d | mod_opcode: %d | scale: %f", delay, fps, mod_opcode, scale);
244 | ConfigValue current_cfg(delay, fps, mod_opcode, scale);
245 | std::thread([=]() {
246 | FPSLimiter::Start(current_cfg);
247 | }).detach();
248 | }
249 | }
250 |
251 | JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved) {
252 | if (reserved) {
253 | ConfigValue* config = reinterpret_cast(reserved);
254 |
255 | std::thread([=]() {
256 | FPSLimiter::Start(*config);
257 | }).detach();
258 | }
259 | return JNI_VERSION_1_6;
260 | }
261 |
262 | #endif
--------------------------------------------------------------------------------
/UnityFPSUnlocker/main.hh:
--------------------------------------------------------------------------------
1 | #ifndef MAIN_HEADER
2 | #define MAIN_HEADER
3 |
4 | #include "third/zygisk.hh"
5 |
6 | #include "utility/config.hh"
7 | #include "utility/logger.hh"
8 |
9 | using zygisk::Api;
10 | using zygisk::AppSpecializeArgs;
11 |
12 | class MyModule : public zygisk::ModuleBase {
13 | public:
14 | virtual void onLoad(Api* api, JNIEnv* env) override final;
15 | virtual void preAppSpecialize(AppSpecializeArgs* args) override final;
16 | virtual void postAppSpecialize(const AppSpecializeArgs* args) override final;
17 |
18 | private:
19 | Api* api;
20 | JNIEnv* env;
21 | int module_dir_fd_ = -1;
22 | int has_custom_cfg_ = false;
23 | const char* package_name_ = nullptr;
24 | ConfigValue current_cfg_;
25 |
26 | void ForHoudini();
27 | };
28 |
29 | #endif // main.hh
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/cursorstreamwrapper.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_CURSORSTREAMWRAPPER_H_
16 | #define RAPIDJSON_CURSORSTREAMWRAPPER_H_
17 |
18 | #include "stream.h"
19 |
20 | #if defined(__GNUC__)
21 | RAPIDJSON_DIAG_PUSH
22 | RAPIDJSON_DIAG_OFF(effc++)
23 | #endif
24 |
25 | #if defined(_MSC_VER) && _MSC_VER <= 1800
26 | RAPIDJSON_DIAG_PUSH
27 | RAPIDJSON_DIAG_OFF(4702) // unreachable code
28 | RAPIDJSON_DIAG_OFF(4512) // assignment operator could not be generated
29 | #endif
30 |
31 | RAPIDJSON_NAMESPACE_BEGIN
32 |
33 |
34 | //! Cursor stream wrapper for counting line and column number if error exists.
35 | /*!
36 | \tparam InputStream Any stream that implements Stream Concept
37 | */
38 | template >
39 | class CursorStreamWrapper : public GenericStreamWrapper {
40 | public:
41 | typedef typename Encoding::Ch Ch;
42 |
43 | CursorStreamWrapper(InputStream& is):
44 | GenericStreamWrapper(is), line_(1), col_(0) {}
45 |
46 | // counting line and column number
47 | Ch Take() {
48 | Ch ch = this->is_.Take();
49 | if(ch == '\n') {
50 | line_ ++;
51 | col_ = 0;
52 | } else {
53 | col_ ++;
54 | }
55 | return ch;
56 | }
57 |
58 | //! Get the error line number, if error exists.
59 | size_t GetLine() const { return line_; }
60 | //! Get the error column number, if error exists.
61 | size_t GetColumn() const { return col_; }
62 |
63 | private:
64 | size_t line_; //!< Current Line
65 | size_t col_; //!< Current Column
66 | };
67 |
68 | #if defined(_MSC_VER) && _MSC_VER <= 1800
69 | RAPIDJSON_DIAG_POP
70 | #endif
71 |
72 | #if defined(__GNUC__)
73 | RAPIDJSON_DIAG_POP
74 | #endif
75 |
76 | RAPIDJSON_NAMESPACE_END
77 |
78 | #endif // RAPIDJSON_CURSORSTREAMWRAPPER_H_
79 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/error/en.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_ERROR_EN_H_
16 | #define RAPIDJSON_ERROR_EN_H_
17 |
18 | #include "error.h"
19 |
20 | #ifdef __clang__
21 | RAPIDJSON_DIAG_PUSH
22 | RAPIDJSON_DIAG_OFF(switch-enum)
23 | RAPIDJSON_DIAG_OFF(covered-switch-default)
24 | #endif
25 |
26 | RAPIDJSON_NAMESPACE_BEGIN
27 |
28 | //! Maps error code of parsing into error message.
29 | /*!
30 | \ingroup RAPIDJSON_ERRORS
31 | \param parseErrorCode Error code obtained in parsing.
32 | \return the error message.
33 | \note User can make a copy of this function for localization.
34 | Using switch-case is safer for future modification of error codes.
35 | */
36 | inline const RAPIDJSON_ERROR_CHARTYPE* GetParseError_En(ParseErrorCode parseErrorCode) {
37 | switch (parseErrorCode) {
38 | case kParseErrorNone: return RAPIDJSON_ERROR_STRING("No error.");
39 |
40 | case kParseErrorDocumentEmpty: return RAPIDJSON_ERROR_STRING("The document is empty.");
41 | case kParseErrorDocumentRootNotSingular: return RAPIDJSON_ERROR_STRING("The document root must not be followed by other values.");
42 |
43 | case kParseErrorValueInvalid: return RAPIDJSON_ERROR_STRING("Invalid value.");
44 |
45 | case kParseErrorObjectMissName: return RAPIDJSON_ERROR_STRING("Missing a name for object member.");
46 | case kParseErrorObjectMissColon: return RAPIDJSON_ERROR_STRING("Missing a colon after a name of object member.");
47 | case kParseErrorObjectMissCommaOrCurlyBracket: return RAPIDJSON_ERROR_STRING("Missing a comma or '}' after an object member.");
48 |
49 | case kParseErrorArrayMissCommaOrSquareBracket: return RAPIDJSON_ERROR_STRING("Missing a comma or ']' after an array element.");
50 |
51 | case kParseErrorStringUnicodeEscapeInvalidHex: return RAPIDJSON_ERROR_STRING("Incorrect hex digit after \\u escape in string.");
52 | case kParseErrorStringUnicodeSurrogateInvalid: return RAPIDJSON_ERROR_STRING("The surrogate pair in string is invalid.");
53 | case kParseErrorStringEscapeInvalid: return RAPIDJSON_ERROR_STRING("Invalid escape character in string.");
54 | case kParseErrorStringMissQuotationMark: return RAPIDJSON_ERROR_STRING("Missing a closing quotation mark in string.");
55 | case kParseErrorStringInvalidEncoding: return RAPIDJSON_ERROR_STRING("Invalid encoding in string.");
56 |
57 | case kParseErrorNumberTooBig: return RAPIDJSON_ERROR_STRING("Number too big to be stored in double.");
58 | case kParseErrorNumberMissFraction: return RAPIDJSON_ERROR_STRING("Miss fraction part in number.");
59 | case kParseErrorNumberMissExponent: return RAPIDJSON_ERROR_STRING("Miss exponent in number.");
60 |
61 | case kParseErrorTermination: return RAPIDJSON_ERROR_STRING("Terminate parsing due to Handler error.");
62 | case kParseErrorUnspecificSyntaxError: return RAPIDJSON_ERROR_STRING("Unspecific syntax error.");
63 |
64 | default: return RAPIDJSON_ERROR_STRING("Unknown error.");
65 | }
66 | }
67 |
68 | RAPIDJSON_NAMESPACE_END
69 |
70 | #ifdef __clang__
71 | RAPIDJSON_DIAG_POP
72 | #endif
73 |
74 | #endif // RAPIDJSON_ERROR_EN_H_
75 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/error/error.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_ERROR_ERROR_H_
16 | #define RAPIDJSON_ERROR_ERROR_H_
17 |
18 | #include "../rapidjson.h"
19 |
20 | #ifdef __clang__
21 | RAPIDJSON_DIAG_PUSH
22 | RAPIDJSON_DIAG_OFF(padded)
23 | #endif
24 |
25 | /*! \file error.h */
26 |
27 | /*! \defgroup RAPIDJSON_ERRORS RapidJSON error handling */
28 |
29 | ///////////////////////////////////////////////////////////////////////////////
30 | // RAPIDJSON_ERROR_CHARTYPE
31 |
32 | //! Character type of error messages.
33 | /*! \ingroup RAPIDJSON_ERRORS
34 | The default character type is \c char.
35 | On Windows, user can define this macro as \c TCHAR for supporting both
36 | unicode/non-unicode settings.
37 | */
38 | #ifndef RAPIDJSON_ERROR_CHARTYPE
39 | #define RAPIDJSON_ERROR_CHARTYPE char
40 | #endif
41 |
42 | ///////////////////////////////////////////////////////////////////////////////
43 | // RAPIDJSON_ERROR_STRING
44 |
45 | //! Macro for converting string literial to \ref RAPIDJSON_ERROR_CHARTYPE[].
46 | /*! \ingroup RAPIDJSON_ERRORS
47 | By default this conversion macro does nothing.
48 | On Windows, user can define this macro as \c _T(x) for supporting both
49 | unicode/non-unicode settings.
50 | */
51 | #ifndef RAPIDJSON_ERROR_STRING
52 | #define RAPIDJSON_ERROR_STRING(x) x
53 | #endif
54 |
55 | RAPIDJSON_NAMESPACE_BEGIN
56 |
57 | ///////////////////////////////////////////////////////////////////////////////
58 | // ParseErrorCode
59 |
60 | //! Error code of parsing.
61 | /*! \ingroup RAPIDJSON_ERRORS
62 | \see GenericReader::Parse, GenericReader::GetParseErrorCode
63 | */
64 | enum ParseErrorCode {
65 | kParseErrorNone = 0, //!< No error.
66 |
67 | kParseErrorDocumentEmpty, //!< The document is empty.
68 | kParseErrorDocumentRootNotSingular, //!< The document root must not follow by other values.
69 |
70 | kParseErrorValueInvalid, //!< Invalid value.
71 |
72 | kParseErrorObjectMissName, //!< Missing a name for object member.
73 | kParseErrorObjectMissColon, //!< Missing a colon after a name of object member.
74 | kParseErrorObjectMissCommaOrCurlyBracket, //!< Missing a comma or '}' after an object member.
75 |
76 | kParseErrorArrayMissCommaOrSquareBracket, //!< Missing a comma or ']' after an array element.
77 |
78 | kParseErrorStringUnicodeEscapeInvalidHex, //!< Incorrect hex digit after \\u escape in string.
79 | kParseErrorStringUnicodeSurrogateInvalid, //!< The surrogate pair in string is invalid.
80 | kParseErrorStringEscapeInvalid, //!< Invalid escape character in string.
81 | kParseErrorStringMissQuotationMark, //!< Missing a closing quotation mark in string.
82 | kParseErrorStringInvalidEncoding, //!< Invalid encoding in string.
83 |
84 | kParseErrorNumberTooBig, //!< Number too big to be stored in double.
85 | kParseErrorNumberMissFraction, //!< Miss fraction part in number.
86 | kParseErrorNumberMissExponent, //!< Miss exponent in number.
87 |
88 | kParseErrorTermination, //!< Parsing was terminated.
89 | kParseErrorUnspecificSyntaxError //!< Unspecific syntax error.
90 | };
91 |
92 | //! Result of parsing (wraps ParseErrorCode)
93 | /*!
94 | \ingroup RAPIDJSON_ERRORS
95 | \code
96 | Document doc;
97 | ParseResult ok = doc.Parse("[42]");
98 | if (!ok) {
99 | fprintf(stderr, "JSON parse error: %s (%u)",
100 | GetParseError_En(ok.Code()), ok.Offset());
101 | exit(EXIT_FAILURE);
102 | }
103 | \endcode
104 | \see GenericReader::Parse, GenericDocument::Parse
105 | */
106 | struct ParseResult {
107 | //!! Unspecified boolean type
108 | typedef bool (ParseResult::*BooleanType)() const;
109 | public:
110 | //! Default constructor, no error.
111 | ParseResult() : code_(kParseErrorNone), offset_(0) {}
112 | //! Constructor to set an error.
113 | ParseResult(ParseErrorCode code, size_t offset) : code_(code), offset_(offset) {}
114 |
115 | //! Get the error code.
116 | ParseErrorCode Code() const { return code_; }
117 | //! Get the error offset, if \ref IsError(), 0 otherwise.
118 | size_t Offset() const { return offset_; }
119 |
120 | //! Explicit conversion to \c bool, returns \c true, iff !\ref IsError().
121 | operator BooleanType() const { return !IsError() ? &ParseResult::IsError : NULL; }
122 | //! Whether the result is an error.
123 | bool IsError() const { return code_ != kParseErrorNone; }
124 |
125 | bool operator==(const ParseResult& that) const { return code_ == that.code_; }
126 | bool operator==(ParseErrorCode code) const { return code_ == code; }
127 | friend bool operator==(ParseErrorCode code, const ParseResult & err) { return code == err.code_; }
128 |
129 | bool operator!=(const ParseResult& that) const { return !(*this == that); }
130 | bool operator!=(ParseErrorCode code) const { return !(*this == code); }
131 | friend bool operator!=(ParseErrorCode code, const ParseResult & err) { return err != code; }
132 |
133 | //! Reset error code.
134 | void Clear() { Set(kParseErrorNone); }
135 | //! Update error code and offset.
136 | void Set(ParseErrorCode code, size_t offset = 0) { code_ = code; offset_ = offset; }
137 |
138 | private:
139 | ParseErrorCode code_;
140 | size_t offset_;
141 | };
142 |
143 | //! Function pointer type of GetParseError().
144 | /*! \ingroup RAPIDJSON_ERRORS
145 |
146 | This is the prototype for \c GetParseError_X(), where \c X is a locale.
147 | User can dynamically change locale in runtime, e.g.:
148 | \code
149 | GetParseErrorFunc GetParseError = GetParseError_En; // or whatever
150 | const RAPIDJSON_ERROR_CHARTYPE* s = GetParseError(document.GetParseErrorCode());
151 | \endcode
152 | */
153 | typedef const RAPIDJSON_ERROR_CHARTYPE* (*GetParseErrorFunc)(ParseErrorCode);
154 |
155 | RAPIDJSON_NAMESPACE_END
156 |
157 | #ifdef __clang__
158 | RAPIDJSON_DIAG_POP
159 | #endif
160 |
161 | #endif // RAPIDJSON_ERROR_ERROR_H_
162 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/filereadstream.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_FILEREADSTREAM_H_
16 | #define RAPIDJSON_FILEREADSTREAM_H_
17 |
18 | #include "stream.h"
19 | #include
20 |
21 | #ifdef __clang__
22 | RAPIDJSON_DIAG_PUSH
23 | RAPIDJSON_DIAG_OFF(padded)
24 | RAPIDJSON_DIAG_OFF(unreachable-code)
25 | RAPIDJSON_DIAG_OFF(missing-noreturn)
26 | #endif
27 |
28 | RAPIDJSON_NAMESPACE_BEGIN
29 |
30 | //! File byte stream for input using fread().
31 | /*!
32 | \note implements Stream concept
33 | */
34 | class FileReadStream {
35 | public:
36 | typedef char Ch; //!< Character type (byte).
37 |
38 | //! Constructor.
39 | /*!
40 | \param fp File pointer opened for read.
41 | \param buffer user-supplied buffer.
42 | \param bufferSize size of buffer in bytes. Must >=4 bytes.
43 | */
44 | FileReadStream(std::FILE* fp, char* buffer, size_t bufferSize) : fp_(fp), buffer_(buffer), bufferSize_(bufferSize), bufferLast_(0), current_(buffer_), readCount_(0), count_(0), eof_(false) {
45 | RAPIDJSON_ASSERT(fp_ != 0);
46 | RAPIDJSON_ASSERT(bufferSize >= 4);
47 | Read();
48 | }
49 |
50 | Ch Peek() const { return *current_; }
51 | Ch Take() { Ch c = *current_; Read(); return c; }
52 | size_t Tell() const { return count_ + static_cast(current_ - buffer_); }
53 |
54 | // Not implemented
55 | void Put(Ch) { RAPIDJSON_ASSERT(false); }
56 | void Flush() { RAPIDJSON_ASSERT(false); }
57 | Ch* PutBegin() { RAPIDJSON_ASSERT(false); return 0; }
58 | size_t PutEnd(Ch*) { RAPIDJSON_ASSERT(false); return 0; }
59 |
60 | // For encoding detection only.
61 | const Ch* Peek4() const {
62 | return (current_ + 4 - !eof_ <= bufferLast_) ? current_ : 0;
63 | }
64 |
65 | private:
66 | void Read() {
67 | if (current_ < bufferLast_)
68 | ++current_;
69 | else if (!eof_) {
70 | count_ += readCount_;
71 | readCount_ = std::fread(buffer_, 1, bufferSize_, fp_);
72 | bufferLast_ = buffer_ + readCount_ - 1;
73 | current_ = buffer_;
74 |
75 | if (readCount_ < bufferSize_) {
76 | buffer_[readCount_] = '\0';
77 | ++bufferLast_;
78 | eof_ = true;
79 | }
80 | }
81 | }
82 |
83 | std::FILE* fp_;
84 | Ch *buffer_;
85 | size_t bufferSize_;
86 | Ch *bufferLast_;
87 | Ch *current_;
88 | size_t readCount_;
89 | size_t count_; //!< Number of characters read
90 | bool eof_;
91 | };
92 |
93 | RAPIDJSON_NAMESPACE_END
94 |
95 | #ifdef __clang__
96 | RAPIDJSON_DIAG_POP
97 | #endif
98 |
99 | #endif // RAPIDJSON_FILESTREAM_H_
100 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/filewritestream.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_FILEWRITESTREAM_H_
16 | #define RAPIDJSON_FILEWRITESTREAM_H_
17 |
18 | #include "stream.h"
19 | #include
20 |
21 | #ifdef __clang__
22 | RAPIDJSON_DIAG_PUSH
23 | RAPIDJSON_DIAG_OFF(unreachable-code)
24 | #endif
25 |
26 | RAPIDJSON_NAMESPACE_BEGIN
27 |
28 | //! Wrapper of C file stream for output using fwrite().
29 | /*!
30 | \note implements Stream concept
31 | */
32 | class FileWriteStream {
33 | public:
34 | typedef char Ch; //!< Character type. Only support char.
35 |
36 | FileWriteStream(std::FILE* fp, char* buffer, size_t bufferSize) : fp_(fp), buffer_(buffer), bufferEnd_(buffer + bufferSize), current_(buffer_) {
37 | RAPIDJSON_ASSERT(fp_ != 0);
38 | }
39 |
40 | void Put(char c) {
41 | if (current_ >= bufferEnd_)
42 | Flush();
43 |
44 | *current_++ = c;
45 | }
46 |
47 | void PutN(char c, size_t n) {
48 | size_t avail = static_cast(bufferEnd_ - current_);
49 | while (n > avail) {
50 | std::memset(current_, c, avail);
51 | current_ += avail;
52 | Flush();
53 | n -= avail;
54 | avail = static_cast(bufferEnd_ - current_);
55 | }
56 |
57 | if (n > 0) {
58 | std::memset(current_, c, n);
59 | current_ += n;
60 | }
61 | }
62 |
63 | void Flush() {
64 | if (current_ != buffer_) {
65 | size_t result = std::fwrite(buffer_, 1, static_cast(current_ - buffer_), fp_);
66 | if (result < static_cast(current_ - buffer_)) {
67 | // failure deliberately ignored at this time
68 | // added to avoid warn_unused_result build errors
69 | }
70 | current_ = buffer_;
71 | }
72 | }
73 |
74 | // Not implemented
75 | char Peek() const { RAPIDJSON_ASSERT(false); return 0; }
76 | char Take() { RAPIDJSON_ASSERT(false); return 0; }
77 | size_t Tell() const { RAPIDJSON_ASSERT(false); return 0; }
78 | char* PutBegin() { RAPIDJSON_ASSERT(false); return 0; }
79 | size_t PutEnd(char*) { RAPIDJSON_ASSERT(false); return 0; }
80 |
81 | private:
82 | // Prohibit copy constructor & assignment operator.
83 | FileWriteStream(const FileWriteStream&);
84 | FileWriteStream& operator=(const FileWriteStream&);
85 |
86 | std::FILE* fp_;
87 | char *buffer_;
88 | char *bufferEnd_;
89 | char *current_;
90 | };
91 |
92 | //! Implement specialized version of PutN() with memset() for better performance.
93 | template<>
94 | inline void PutN(FileWriteStream& stream, char c, size_t n) {
95 | stream.PutN(c, n);
96 | }
97 |
98 | RAPIDJSON_NAMESPACE_END
99 |
100 | #ifdef __clang__
101 | RAPIDJSON_DIAG_POP
102 | #endif
103 |
104 | #endif // RAPIDJSON_FILESTREAM_H_
105 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/fwd.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_FWD_H_
16 | #define RAPIDJSON_FWD_H_
17 |
18 | #include "rapidjson.h"
19 |
20 | RAPIDJSON_NAMESPACE_BEGIN
21 |
22 | // encodings.h
23 |
24 | template struct UTF8;
25 | template struct UTF16;
26 | template struct UTF16BE;
27 | template struct UTF16LE;
28 | template struct UTF32;
29 | template struct UTF32BE;
30 | template struct UTF32LE;
31 | template struct ASCII;
32 | template struct AutoUTF;
33 |
34 | template
35 | struct Transcoder;
36 |
37 | // allocators.h
38 |
39 | class CrtAllocator;
40 |
41 | template
42 | class MemoryPoolAllocator;
43 |
44 | // stream.h
45 |
46 | template
47 | struct GenericStringStream;
48 |
49 | typedef GenericStringStream > StringStream;
50 |
51 | template
52 | struct GenericInsituStringStream;
53 |
54 | typedef GenericInsituStringStream > InsituStringStream;
55 |
56 | // stringbuffer.h
57 |
58 | template
59 | class GenericStringBuffer;
60 |
61 | typedef GenericStringBuffer, CrtAllocator> StringBuffer;
62 |
63 | // filereadstream.h
64 |
65 | class FileReadStream;
66 |
67 | // filewritestream.h
68 |
69 | class FileWriteStream;
70 |
71 | // memorybuffer.h
72 |
73 | template
74 | struct GenericMemoryBuffer;
75 |
76 | typedef GenericMemoryBuffer MemoryBuffer;
77 |
78 | // memorystream.h
79 |
80 | struct MemoryStream;
81 |
82 | // reader.h
83 |
84 | template
85 | struct BaseReaderHandler;
86 |
87 | template
88 | class GenericReader;
89 |
90 | typedef GenericReader, UTF8, CrtAllocator> Reader;
91 |
92 | // writer.h
93 |
94 | template
95 | class Writer;
96 |
97 | // prettywriter.h
98 |
99 | template
100 | class PrettyWriter;
101 |
102 | // document.h
103 |
104 | template
105 | class GenericMember;
106 |
107 | template
108 | class GenericMemberIterator;
109 |
110 | template
111 | struct GenericStringRef;
112 |
113 | template
114 | class GenericValue;
115 |
116 | typedef GenericValue, MemoryPoolAllocator > Value;
117 |
118 | template
119 | class GenericDocument;
120 |
121 | typedef GenericDocument, MemoryPoolAllocator, CrtAllocator> Document;
122 |
123 | // pointer.h
124 |
125 | template
126 | class GenericPointer;
127 |
128 | typedef GenericPointer Pointer;
129 |
130 | // schema.h
131 |
132 | template
133 | class IGenericRemoteSchemaDocumentProvider;
134 |
135 | template
136 | class GenericSchemaDocument;
137 |
138 | typedef GenericSchemaDocument SchemaDocument;
139 | typedef IGenericRemoteSchemaDocumentProvider IRemoteSchemaDocumentProvider;
140 |
141 | template <
142 | typename SchemaDocumentType,
143 | typename OutputHandler,
144 | typename StateAllocator>
145 | class GenericSchemaValidator;
146 |
147 | typedef GenericSchemaValidator, void>, CrtAllocator> SchemaValidator;
148 |
149 | RAPIDJSON_NAMESPACE_END
150 |
151 | #endif // RAPIDJSON_RAPIDJSONFWD_H_
152 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/internal/clzll.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_CLZLL_H_
16 | #define RAPIDJSON_CLZLL_H_
17 |
18 | #include "../rapidjson.h"
19 |
20 | #if defined(_MSC_VER)
21 | #include
22 | #if defined(_WIN64)
23 | #pragma intrinsic(_BitScanReverse64)
24 | #else
25 | #pragma intrinsic(_BitScanReverse)
26 | #endif
27 | #endif
28 |
29 | RAPIDJSON_NAMESPACE_BEGIN
30 | namespace internal {
31 |
32 | inline uint32_t clzll(uint64_t x) {
33 | // Passing 0 to __builtin_clzll is UB in GCC and results in an
34 | // infinite loop in the software implementation.
35 | RAPIDJSON_ASSERT(x != 0);
36 |
37 | #if defined(_MSC_VER)
38 | unsigned long r = 0;
39 | #if defined(_WIN64)
40 | _BitScanReverse64(&r, x);
41 | #else
42 | // Scan the high 32 bits.
43 | if (_BitScanReverse(&r, static_cast(x >> 32)))
44 | return 63 - (r + 32);
45 |
46 | // Scan the low 32 bits.
47 | _BitScanReverse(&r, static_cast(x & 0xFFFFFFFF));
48 | #endif // _WIN64
49 |
50 | return 63 - r;
51 | #elif (defined(__GNUC__) && __GNUC__ >= 4) || RAPIDJSON_HAS_BUILTIN(__builtin_clzll)
52 | // __builtin_clzll wrapper
53 | return static_cast(__builtin_clzll(x));
54 | #else
55 | // naive version
56 | uint32_t r;
57 | while (!(x & (static_cast(1) << 63))) {
58 | x <<= 1;
59 | ++r;
60 | }
61 |
62 | return r;
63 | #endif // _MSC_VER
64 | }
65 |
66 | #define RAPIDJSON_CLZLL RAPIDJSON_NAMESPACE::internal::clzll
67 |
68 | } // namespace internal
69 | RAPIDJSON_NAMESPACE_END
70 |
71 | #endif // RAPIDJSON_CLZLL_H_
72 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/internal/dtoa.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | // This is a C++ header-only implementation of Grisu2 algorithm from the publication:
16 | // Loitsch, Florian. "Printing floating-point numbers quickly and accurately with
17 | // integers." ACM Sigplan Notices 45.6 (2010): 233-243.
18 |
19 | #ifndef RAPIDJSON_DTOA_
20 | #define RAPIDJSON_DTOA_
21 |
22 | #include "itoa.h" // GetDigitsLut()
23 | #include "diyfp.h"
24 | #include "ieee754.h"
25 |
26 | RAPIDJSON_NAMESPACE_BEGIN
27 | namespace internal {
28 |
29 | #ifdef __GNUC__
30 | RAPIDJSON_DIAG_PUSH
31 | RAPIDJSON_DIAG_OFF(effc++)
32 | RAPIDJSON_DIAG_OFF(array-bounds) // some gcc versions generate wrong warnings https://gcc.gnu.org/bugzilla/show_bug.cgi?id=59124
33 | #endif
34 |
35 | inline void GrisuRound(char* buffer, int len, uint64_t delta, uint64_t rest, uint64_t ten_kappa, uint64_t wp_w) {
36 | while (rest < wp_w && delta - rest >= ten_kappa &&
37 | (rest + ten_kappa < wp_w || /// closer
38 | wp_w - rest > rest + ten_kappa - wp_w)) {
39 | buffer[len - 1]--;
40 | rest += ten_kappa;
41 | }
42 | }
43 |
44 | inline int CountDecimalDigit32(uint32_t n) {
45 | // Simple pure C++ implementation was faster than __builtin_clz version in this situation.
46 | if (n < 10) return 1;
47 | if (n < 100) return 2;
48 | if (n < 1000) return 3;
49 | if (n < 10000) return 4;
50 | if (n < 100000) return 5;
51 | if (n < 1000000) return 6;
52 | if (n < 10000000) return 7;
53 | if (n < 100000000) return 8;
54 | // Will not reach 10 digits in DigitGen()
55 | //if (n < 1000000000) return 9;
56 | //return 10;
57 | return 9;
58 | }
59 |
60 | inline void DigitGen(const DiyFp& W, const DiyFp& Mp, uint64_t delta, char* buffer, int* len, int* K) {
61 | static const uint32_t kPow10[] = { 1, 10, 100, 1000, 10000, 100000, 1000000, 10000000, 100000000, 1000000000 };
62 | const DiyFp one(uint64_t(1) << -Mp.e, Mp.e);
63 | const DiyFp wp_w = Mp - W;
64 | uint32_t p1 = static_cast(Mp.f >> -one.e);
65 | uint64_t p2 = Mp.f & (one.f - 1);
66 | int kappa = CountDecimalDigit32(p1); // kappa in [0, 9]
67 | *len = 0;
68 |
69 | while (kappa > 0) {
70 | uint32_t d = 0;
71 | switch (kappa) {
72 | case 9: d = p1 / 100000000; p1 %= 100000000; break;
73 | case 8: d = p1 / 10000000; p1 %= 10000000; break;
74 | case 7: d = p1 / 1000000; p1 %= 1000000; break;
75 | case 6: d = p1 / 100000; p1 %= 100000; break;
76 | case 5: d = p1 / 10000; p1 %= 10000; break;
77 | case 4: d = p1 / 1000; p1 %= 1000; break;
78 | case 3: d = p1 / 100; p1 %= 100; break;
79 | case 2: d = p1 / 10; p1 %= 10; break;
80 | case 1: d = p1; p1 = 0; break;
81 | default:;
82 | }
83 | if (d || *len)
84 | buffer[(*len)++] = static_cast('0' + static_cast(d));
85 | kappa--;
86 | uint64_t tmp = (static_cast(p1) << -one.e) + p2;
87 | if (tmp <= delta) {
88 | *K += kappa;
89 | GrisuRound(buffer, *len, delta, tmp, static_cast(kPow10[kappa]) << -one.e, wp_w.f);
90 | return;
91 | }
92 | }
93 |
94 | // kappa = 0
95 | for (;;) {
96 | p2 *= 10;
97 | delta *= 10;
98 | char d = static_cast(p2 >> -one.e);
99 | if (d || *len)
100 | buffer[(*len)++] = static_cast('0' + d);
101 | p2 &= one.f - 1;
102 | kappa--;
103 | if (p2 < delta) {
104 | *K += kappa;
105 | int index = -kappa;
106 | GrisuRound(buffer, *len, delta, p2, one.f, wp_w.f * (index < 9 ? kPow10[index] : 0));
107 | return;
108 | }
109 | }
110 | }
111 |
112 | inline void Grisu2(double value, char* buffer, int* length, int* K) {
113 | const DiyFp v(value);
114 | DiyFp w_m, w_p;
115 | v.NormalizedBoundaries(&w_m, &w_p);
116 |
117 | const DiyFp c_mk = GetCachedPower(w_p.e, K);
118 | const DiyFp W = v.Normalize() * c_mk;
119 | DiyFp Wp = w_p * c_mk;
120 | DiyFp Wm = w_m * c_mk;
121 | Wm.f++;
122 | Wp.f--;
123 | DigitGen(W, Wp, Wp.f - Wm.f, buffer, length, K);
124 | }
125 |
126 | inline char* WriteExponent(int K, char* buffer) {
127 | if (K < 0) {
128 | *buffer++ = '-';
129 | K = -K;
130 | }
131 |
132 | if (K >= 100) {
133 | *buffer++ = static_cast('0' + static_cast(K / 100));
134 | K %= 100;
135 | const char* d = GetDigitsLut() + K * 2;
136 | *buffer++ = d[0];
137 | *buffer++ = d[1];
138 | }
139 | else if (K >= 10) {
140 | const char* d = GetDigitsLut() + K * 2;
141 | *buffer++ = d[0];
142 | *buffer++ = d[1];
143 | }
144 | else
145 | *buffer++ = static_cast('0' + static_cast(K));
146 |
147 | return buffer;
148 | }
149 |
150 | inline char* Prettify(char* buffer, int length, int k, int maxDecimalPlaces) {
151 | const int kk = length + k; // 10^(kk-1) <= v < 10^kk
152 |
153 | if (0 <= k && kk <= 21) {
154 | // 1234e7 -> 12340000000
155 | for (int i = length; i < kk; i++)
156 | buffer[i] = '0';
157 | buffer[kk] = '.';
158 | buffer[kk + 1] = '0';
159 | return &buffer[kk + 2];
160 | }
161 | else if (0 < kk && kk <= 21) {
162 | // 1234e-2 -> 12.34
163 | std::memmove(&buffer[kk + 1], &buffer[kk], static_cast(length - kk));
164 | buffer[kk] = '.';
165 | if (0 > k + maxDecimalPlaces) {
166 | // When maxDecimalPlaces = 2, 1.2345 -> 1.23, 1.102 -> 1.1
167 | // Remove extra trailing zeros (at least one) after truncation.
168 | for (int i = kk + maxDecimalPlaces; i > kk + 1; i--)
169 | if (buffer[i] != '0')
170 | return &buffer[i + 1];
171 | return &buffer[kk + 2]; // Reserve one zero
172 | }
173 | else
174 | return &buffer[length + 1];
175 | }
176 | else if (-6 < kk && kk <= 0) {
177 | // 1234e-6 -> 0.001234
178 | const int offset = 2 - kk;
179 | std::memmove(&buffer[offset], &buffer[0], static_cast(length));
180 | buffer[0] = '0';
181 | buffer[1] = '.';
182 | for (int i = 2; i < offset; i++)
183 | buffer[i] = '0';
184 | if (length - kk > maxDecimalPlaces) {
185 | // When maxDecimalPlaces = 2, 0.123 -> 0.12, 0.102 -> 0.1
186 | // Remove extra trailing zeros (at least one) after truncation.
187 | for (int i = maxDecimalPlaces + 1; i > 2; i--)
188 | if (buffer[i] != '0')
189 | return &buffer[i + 1];
190 | return &buffer[3]; // Reserve one zero
191 | }
192 | else
193 | return &buffer[length + offset];
194 | }
195 | else if (kk < -maxDecimalPlaces) {
196 | // Truncate to zero
197 | buffer[0] = '0';
198 | buffer[1] = '.';
199 | buffer[2] = '0';
200 | return &buffer[3];
201 | }
202 | else if (length == 1) {
203 | // 1e30
204 | buffer[1] = 'e';
205 | return WriteExponent(kk - 1, &buffer[2]);
206 | }
207 | else {
208 | // 1234e30 -> 1.234e33
209 | std::memmove(&buffer[2], &buffer[1], static_cast(length - 1));
210 | buffer[1] = '.';
211 | buffer[length + 1] = 'e';
212 | return WriteExponent(kk - 1, &buffer[0 + length + 2]);
213 | }
214 | }
215 |
216 | inline char* dtoa(double value, char* buffer, int maxDecimalPlaces = 324) {
217 | RAPIDJSON_ASSERT(maxDecimalPlaces >= 1);
218 | Double d(value);
219 | if (d.IsZero()) {
220 | if (d.Sign())
221 | *buffer++ = '-'; // -0.0, Issue #289
222 | buffer[0] = '0';
223 | buffer[1] = '.';
224 | buffer[2] = '0';
225 | return &buffer[3];
226 | }
227 | else {
228 | if (value < 0) {
229 | *buffer++ = '-';
230 | value = -value;
231 | }
232 | int length, K;
233 | Grisu2(value, buffer, &length, &K);
234 | return Prettify(buffer, length, K, maxDecimalPlaces);
235 | }
236 | }
237 |
238 | #ifdef __GNUC__
239 | RAPIDJSON_DIAG_POP
240 | #endif
241 |
242 | } // namespace internal
243 | RAPIDJSON_NAMESPACE_END
244 |
245 | #endif // RAPIDJSON_DTOA_
246 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/internal/ieee754.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_IEEE754_
16 | #define RAPIDJSON_IEEE754_
17 |
18 | #include "../rapidjson.h"
19 |
20 | RAPIDJSON_NAMESPACE_BEGIN
21 | namespace internal {
22 |
23 | class Double {
24 | public:
25 | Double() {}
26 | Double(double d) : d_(d) {}
27 | Double(uint64_t u) : u_(u) {}
28 |
29 | double Value() const { return d_; }
30 | uint64_t Uint64Value() const { return u_; }
31 |
32 | double NextPositiveDouble() const {
33 | RAPIDJSON_ASSERT(!Sign());
34 | return Double(u_ + 1).Value();
35 | }
36 |
37 | bool Sign() const { return (u_ & kSignMask) != 0; }
38 | uint64_t Significand() const { return u_ & kSignificandMask; }
39 | int Exponent() const { return static_cast(((u_ & kExponentMask) >> kSignificandSize) - kExponentBias); }
40 |
41 | bool IsNan() const { return (u_ & kExponentMask) == kExponentMask && Significand() != 0; }
42 | bool IsInf() const { return (u_ & kExponentMask) == kExponentMask && Significand() == 0; }
43 | bool IsNanOrInf() const { return (u_ & kExponentMask) == kExponentMask; }
44 | bool IsNormal() const { return (u_ & kExponentMask) != 0 || Significand() == 0; }
45 | bool IsZero() const { return (u_ & (kExponentMask | kSignificandMask)) == 0; }
46 |
47 | uint64_t IntegerSignificand() const { return IsNormal() ? Significand() | kHiddenBit : Significand(); }
48 | int IntegerExponent() const { return (IsNormal() ? Exponent() : kDenormalExponent) - kSignificandSize; }
49 | uint64_t ToBias() const { return (u_ & kSignMask) ? ~u_ + 1 : u_ | kSignMask; }
50 |
51 | static int EffectiveSignificandSize(int order) {
52 | if (order >= -1021)
53 | return 53;
54 | else if (order <= -1074)
55 | return 0;
56 | else
57 | return order + 1074;
58 | }
59 |
60 | private:
61 | static const int kSignificandSize = 52;
62 | static const int kExponentBias = 0x3FF;
63 | static const int kDenormalExponent = 1 - kExponentBias;
64 | static const uint64_t kSignMask = RAPIDJSON_UINT64_C2(0x80000000, 0x00000000);
65 | static const uint64_t kExponentMask = RAPIDJSON_UINT64_C2(0x7FF00000, 0x00000000);
66 | static const uint64_t kSignificandMask = RAPIDJSON_UINT64_C2(0x000FFFFF, 0xFFFFFFFF);
67 | static const uint64_t kHiddenBit = RAPIDJSON_UINT64_C2(0x00100000, 0x00000000);
68 |
69 | union {
70 | double d_;
71 | uint64_t u_;
72 | };
73 | };
74 |
75 | } // namespace internal
76 | RAPIDJSON_NAMESPACE_END
77 |
78 | #endif // RAPIDJSON_IEEE754_
79 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/internal/meta.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_INTERNAL_META_H_
16 | #define RAPIDJSON_INTERNAL_META_H_
17 |
18 | #include "../rapidjson.h"
19 |
20 | #ifdef __GNUC__
21 | RAPIDJSON_DIAG_PUSH
22 | RAPIDJSON_DIAG_OFF(effc++)
23 | #endif
24 |
25 | #if defined(_MSC_VER) && !defined(__clang__)
26 | RAPIDJSON_DIAG_PUSH
27 | RAPIDJSON_DIAG_OFF(6334)
28 | #endif
29 |
30 | #if RAPIDJSON_HAS_CXX11_TYPETRAITS
31 | #include
32 | #endif
33 |
34 | //@cond RAPIDJSON_INTERNAL
35 | RAPIDJSON_NAMESPACE_BEGIN
36 | namespace internal {
37 |
38 | // Helper to wrap/convert arbitrary types to void, useful for arbitrary type matching
39 | template struct Void { typedef void Type; };
40 |
41 | ///////////////////////////////////////////////////////////////////////////////
42 | // BoolType, TrueType, FalseType
43 | //
44 | template struct BoolType {
45 | static const bool Value = Cond;
46 | typedef BoolType Type;
47 | };
48 | typedef BoolType TrueType;
49 | typedef BoolType FalseType;
50 |
51 |
52 | ///////////////////////////////////////////////////////////////////////////////
53 | // SelectIf, BoolExpr, NotExpr, AndExpr, OrExpr
54 | //
55 |
56 | template struct SelectIfImpl { template struct Apply { typedef T1 Type; }; };
57 | template <> struct SelectIfImpl { template struct Apply { typedef T2 Type; }; };
58 | template struct SelectIfCond : SelectIfImpl::template Apply {};
59 | template struct SelectIf : SelectIfCond {};
60 |
61 | template struct AndExprCond : FalseType {};
62 | template <> struct AndExprCond : TrueType {};
63 | template struct OrExprCond : TrueType {};
64 | template <> struct OrExprCond : FalseType {};
65 |
66 | template struct BoolExpr : SelectIf::Type {};
67 | template struct NotExpr : SelectIf::Type {};
68 | template struct AndExpr : AndExprCond::Type {};
69 | template struct OrExpr : OrExprCond::Type {};
70 |
71 |
72 | ///////////////////////////////////////////////////////////////////////////////
73 | // AddConst, MaybeAddConst, RemoveConst
74 | template struct AddConst { typedef const T Type; };
75 | template struct MaybeAddConst : SelectIfCond {};
76 | template struct RemoveConst { typedef T Type; };
77 | template struct RemoveConst { typedef T Type; };
78 |
79 |
80 | ///////////////////////////////////////////////////////////////////////////////
81 | // IsSame, IsConst, IsMoreConst, IsPointer
82 | //
83 | template struct IsSame : FalseType {};
84 | template struct IsSame : TrueType {};
85 |
86 | template struct IsConst : FalseType {};
87 | template struct IsConst : TrueType {};
88 |
89 | template
90 | struct IsMoreConst
91 | : AndExpr::Type, typename RemoveConst::Type>,
92 | BoolType::Value >= IsConst::Value> >::Type {};
93 |
94 | template struct IsPointer : FalseType {};
95 | template struct IsPointer : TrueType {};
96 |
97 | ///////////////////////////////////////////////////////////////////////////////
98 | // IsBaseOf
99 | //
100 | #if RAPIDJSON_HAS_CXX11_TYPETRAITS
101 |
102 | template struct IsBaseOf
103 | : BoolType< ::std::is_base_of::value> {};
104 |
105 | #else // simplified version adopted from Boost
106 |
107 | template struct IsBaseOfImpl {
108 | RAPIDJSON_STATIC_ASSERT(sizeof(B) != 0);
109 | RAPIDJSON_STATIC_ASSERT(sizeof(D) != 0);
110 |
111 | typedef char (&Yes)[1];
112 | typedef char (&No) [2];
113 |
114 | template
115 | static Yes Check(const D*, T);
116 | static No Check(const B*, int);
117 |
118 | struct Host {
119 | operator const B*() const;
120 | operator const D*();
121 | };
122 |
123 | enum { Value = (sizeof(Check(Host(), 0)) == sizeof(Yes)) };
124 | };
125 |
126 | template struct IsBaseOf
127 | : OrExpr, BoolExpr > >::Type {};
128 |
129 | #endif // RAPIDJSON_HAS_CXX11_TYPETRAITS
130 |
131 |
132 | //////////////////////////////////////////////////////////////////////////
133 | // EnableIf / DisableIf
134 | //
135 | template struct EnableIfCond { typedef T Type; };
136 | template struct EnableIfCond { /* empty */ };
137 |
138 | template struct DisableIfCond { typedef T Type; };
139 | template struct DisableIfCond { /* empty */ };
140 |
141 | template
142 | struct EnableIf : EnableIfCond {};
143 |
144 | template
145 | struct DisableIf : DisableIfCond {};
146 |
147 | // SFINAE helpers
148 | struct SfinaeTag {};
149 | template struct RemoveSfinaeTag;
150 | template struct RemoveSfinaeTag { typedef T Type; };
151 |
152 | #define RAPIDJSON_REMOVEFPTR_(type) \
153 | typename ::RAPIDJSON_NAMESPACE::internal::RemoveSfinaeTag \
154 | < ::RAPIDJSON_NAMESPACE::internal::SfinaeTag&(*) type>::Type
155 |
156 | #define RAPIDJSON_ENABLEIF(cond) \
157 | typename ::RAPIDJSON_NAMESPACE::internal::EnableIf \
158 | ::Type * = NULL
159 |
160 | #define RAPIDJSON_DISABLEIF(cond) \
161 | typename ::RAPIDJSON_NAMESPACE::internal::DisableIf \
162 | ::Type * = NULL
163 |
164 | #define RAPIDJSON_ENABLEIF_RETURN(cond,returntype) \
165 | typename ::RAPIDJSON_NAMESPACE::internal::EnableIf \
166 | ::Type
168 |
169 | #define RAPIDJSON_DISABLEIF_RETURN(cond,returntype) \
170 | typename ::RAPIDJSON_NAMESPACE::internal::DisableIf \
171 | ::Type
173 |
174 | } // namespace internal
175 | RAPIDJSON_NAMESPACE_END
176 | //@endcond
177 |
178 | #if defined(_MSC_VER) && !defined(__clang__)
179 | RAPIDJSON_DIAG_POP
180 | #endif
181 |
182 | #ifdef __GNUC__
183 | RAPIDJSON_DIAG_POP
184 | #endif
185 |
186 | #endif // RAPIDJSON_INTERNAL_META_H_
187 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/internal/pow10.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_POW10_
16 | #define RAPIDJSON_POW10_
17 |
18 | #include "../rapidjson.h"
19 |
20 | RAPIDJSON_NAMESPACE_BEGIN
21 | namespace internal {
22 |
23 | //! Computes integer powers of 10 in double (10.0^n).
24 | /*! This function uses lookup table for fast and accurate results.
25 | \param n non-negative exponent. Must <= 308.
26 | \return 10.0^n
27 | */
28 | inline double Pow10(int n) {
29 | static const double e[] = { // 1e-0...1e308: 309 * 8 bytes = 2472 bytes
30 | 1e+0,
31 | 1e+1, 1e+2, 1e+3, 1e+4, 1e+5, 1e+6, 1e+7, 1e+8, 1e+9, 1e+10, 1e+11, 1e+12, 1e+13, 1e+14, 1e+15, 1e+16, 1e+17, 1e+18, 1e+19, 1e+20,
32 | 1e+21, 1e+22, 1e+23, 1e+24, 1e+25, 1e+26, 1e+27, 1e+28, 1e+29, 1e+30, 1e+31, 1e+32, 1e+33, 1e+34, 1e+35, 1e+36, 1e+37, 1e+38, 1e+39, 1e+40,
33 | 1e+41, 1e+42, 1e+43, 1e+44, 1e+45, 1e+46, 1e+47, 1e+48, 1e+49, 1e+50, 1e+51, 1e+52, 1e+53, 1e+54, 1e+55, 1e+56, 1e+57, 1e+58, 1e+59, 1e+60,
34 | 1e+61, 1e+62, 1e+63, 1e+64, 1e+65, 1e+66, 1e+67, 1e+68, 1e+69, 1e+70, 1e+71, 1e+72, 1e+73, 1e+74, 1e+75, 1e+76, 1e+77, 1e+78, 1e+79, 1e+80,
35 | 1e+81, 1e+82, 1e+83, 1e+84, 1e+85, 1e+86, 1e+87, 1e+88, 1e+89, 1e+90, 1e+91, 1e+92, 1e+93, 1e+94, 1e+95, 1e+96, 1e+97, 1e+98, 1e+99, 1e+100,
36 | 1e+101,1e+102,1e+103,1e+104,1e+105,1e+106,1e+107,1e+108,1e+109,1e+110,1e+111,1e+112,1e+113,1e+114,1e+115,1e+116,1e+117,1e+118,1e+119,1e+120,
37 | 1e+121,1e+122,1e+123,1e+124,1e+125,1e+126,1e+127,1e+128,1e+129,1e+130,1e+131,1e+132,1e+133,1e+134,1e+135,1e+136,1e+137,1e+138,1e+139,1e+140,
38 | 1e+141,1e+142,1e+143,1e+144,1e+145,1e+146,1e+147,1e+148,1e+149,1e+150,1e+151,1e+152,1e+153,1e+154,1e+155,1e+156,1e+157,1e+158,1e+159,1e+160,
39 | 1e+161,1e+162,1e+163,1e+164,1e+165,1e+166,1e+167,1e+168,1e+169,1e+170,1e+171,1e+172,1e+173,1e+174,1e+175,1e+176,1e+177,1e+178,1e+179,1e+180,
40 | 1e+181,1e+182,1e+183,1e+184,1e+185,1e+186,1e+187,1e+188,1e+189,1e+190,1e+191,1e+192,1e+193,1e+194,1e+195,1e+196,1e+197,1e+198,1e+199,1e+200,
41 | 1e+201,1e+202,1e+203,1e+204,1e+205,1e+206,1e+207,1e+208,1e+209,1e+210,1e+211,1e+212,1e+213,1e+214,1e+215,1e+216,1e+217,1e+218,1e+219,1e+220,
42 | 1e+221,1e+222,1e+223,1e+224,1e+225,1e+226,1e+227,1e+228,1e+229,1e+230,1e+231,1e+232,1e+233,1e+234,1e+235,1e+236,1e+237,1e+238,1e+239,1e+240,
43 | 1e+241,1e+242,1e+243,1e+244,1e+245,1e+246,1e+247,1e+248,1e+249,1e+250,1e+251,1e+252,1e+253,1e+254,1e+255,1e+256,1e+257,1e+258,1e+259,1e+260,
44 | 1e+261,1e+262,1e+263,1e+264,1e+265,1e+266,1e+267,1e+268,1e+269,1e+270,1e+271,1e+272,1e+273,1e+274,1e+275,1e+276,1e+277,1e+278,1e+279,1e+280,
45 | 1e+281,1e+282,1e+283,1e+284,1e+285,1e+286,1e+287,1e+288,1e+289,1e+290,1e+291,1e+292,1e+293,1e+294,1e+295,1e+296,1e+297,1e+298,1e+299,1e+300,
46 | 1e+301,1e+302,1e+303,1e+304,1e+305,1e+306,1e+307,1e+308
47 | };
48 | RAPIDJSON_ASSERT(n >= 0 && n <= 308);
49 | return e[n];
50 | }
51 |
52 | } // namespace internal
53 | RAPIDJSON_NAMESPACE_END
54 |
55 | #endif // RAPIDJSON_POW10_
56 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/internal/stack.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_INTERNAL_STACK_H_
16 | #define RAPIDJSON_INTERNAL_STACK_H_
17 |
18 | #include "../allocators.h"
19 | #include "swap.h"
20 | #include
21 |
22 | #if defined(__clang__)
23 | RAPIDJSON_DIAG_PUSH
24 | RAPIDJSON_DIAG_OFF(c++98-compat)
25 | #endif
26 |
27 | RAPIDJSON_NAMESPACE_BEGIN
28 | namespace internal {
29 |
30 | ///////////////////////////////////////////////////////////////////////////////
31 | // Stack
32 |
33 | //! A type-unsafe stack for storing different types of data.
34 | /*! \tparam Allocator Allocator for allocating stack memory.
35 | */
36 | template
37 | class Stack {
38 | public:
39 | // Optimization note: Do not allocate memory for stack_ in constructor.
40 | // Do it lazily when first Push() -> Expand() -> Resize().
41 | Stack(Allocator* allocator, size_t stackCapacity) : allocator_(allocator), ownAllocator_(0), stack_(0), stackTop_(0), stackEnd_(0), initialCapacity_(stackCapacity) {
42 | }
43 |
44 | #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
45 | Stack(Stack&& rhs)
46 | : allocator_(rhs.allocator_),
47 | ownAllocator_(rhs.ownAllocator_),
48 | stack_(rhs.stack_),
49 | stackTop_(rhs.stackTop_),
50 | stackEnd_(rhs.stackEnd_),
51 | initialCapacity_(rhs.initialCapacity_)
52 | {
53 | rhs.allocator_ = 0;
54 | rhs.ownAllocator_ = 0;
55 | rhs.stack_ = 0;
56 | rhs.stackTop_ = 0;
57 | rhs.stackEnd_ = 0;
58 | rhs.initialCapacity_ = 0;
59 | }
60 | #endif
61 |
62 | ~Stack() {
63 | Destroy();
64 | }
65 |
66 | #if RAPIDJSON_HAS_CXX11_RVALUE_REFS
67 | Stack& operator=(Stack&& rhs) {
68 | if (&rhs != this)
69 | {
70 | Destroy();
71 |
72 | allocator_ = rhs.allocator_;
73 | ownAllocator_ = rhs.ownAllocator_;
74 | stack_ = rhs.stack_;
75 | stackTop_ = rhs.stackTop_;
76 | stackEnd_ = rhs.stackEnd_;
77 | initialCapacity_ = rhs.initialCapacity_;
78 |
79 | rhs.allocator_ = 0;
80 | rhs.ownAllocator_ = 0;
81 | rhs.stack_ = 0;
82 | rhs.stackTop_ = 0;
83 | rhs.stackEnd_ = 0;
84 | rhs.initialCapacity_ = 0;
85 | }
86 | return *this;
87 | }
88 | #endif
89 |
90 | void Swap(Stack& rhs) RAPIDJSON_NOEXCEPT {
91 | internal::Swap(allocator_, rhs.allocator_);
92 | internal::Swap(ownAllocator_, rhs.ownAllocator_);
93 | internal::Swap(stack_, rhs.stack_);
94 | internal::Swap(stackTop_, rhs.stackTop_);
95 | internal::Swap(stackEnd_, rhs.stackEnd_);
96 | internal::Swap(initialCapacity_, rhs.initialCapacity_);
97 | }
98 |
99 | void Clear() { stackTop_ = stack_; }
100 |
101 | void ShrinkToFit() {
102 | if (Empty()) {
103 | // If the stack is empty, completely deallocate the memory.
104 | Allocator::Free(stack_); // NOLINT (+clang-analyzer-unix.Malloc)
105 | stack_ = 0;
106 | stackTop_ = 0;
107 | stackEnd_ = 0;
108 | }
109 | else
110 | Resize(GetSize());
111 | }
112 |
113 | // Optimization note: try to minimize the size of this function for force inline.
114 | // Expansion is run very infrequently, so it is moved to another (probably non-inline) function.
115 | template
116 | RAPIDJSON_FORCEINLINE void Reserve(size_t count = 1) {
117 | // Expand the stack if needed
118 | if (RAPIDJSON_UNLIKELY(static_cast(sizeof(T) * count) > (stackEnd_ - stackTop_)))
119 | Expand(count);
120 | }
121 |
122 | template
123 | RAPIDJSON_FORCEINLINE T* Push(size_t count = 1) {
124 | Reserve(count);
125 | return PushUnsafe(count);
126 | }
127 |
128 | template
129 | RAPIDJSON_FORCEINLINE T* PushUnsafe(size_t count = 1) {
130 | RAPIDJSON_ASSERT(stackTop_);
131 | RAPIDJSON_ASSERT(static_cast(sizeof(T) * count) <= (stackEnd_ - stackTop_));
132 | T* ret = reinterpret_cast(stackTop_);
133 | stackTop_ += sizeof(T) * count;
134 | return ret;
135 | }
136 |
137 | template
138 | T* Pop(size_t count) {
139 | RAPIDJSON_ASSERT(GetSize() >= count * sizeof(T));
140 | stackTop_ -= count * sizeof(T);
141 | return reinterpret_cast(stackTop_);
142 | }
143 |
144 | template
145 | T* Top() {
146 | RAPIDJSON_ASSERT(GetSize() >= sizeof(T));
147 | return reinterpret_cast(stackTop_ - sizeof(T));
148 | }
149 |
150 | template
151 | const T* Top() const {
152 | RAPIDJSON_ASSERT(GetSize() >= sizeof(T));
153 | return reinterpret_cast(stackTop_ - sizeof(T));
154 | }
155 |
156 | template
157 | T* End() { return reinterpret_cast(stackTop_); }
158 |
159 | template
160 | const T* End() const { return reinterpret_cast(stackTop_); }
161 |
162 | template
163 | T* Bottom() { return reinterpret_cast(stack_); }
164 |
165 | template
166 | const T* Bottom() const { return reinterpret_cast(stack_); }
167 |
168 | bool HasAllocator() const {
169 | return allocator_ != 0;
170 | }
171 |
172 | Allocator& GetAllocator() {
173 | RAPIDJSON_ASSERT(allocator_);
174 | return *allocator_;
175 | }
176 |
177 | bool Empty() const { return stackTop_ == stack_; }
178 | size_t GetSize() const { return static_cast(stackTop_ - stack_); }
179 | size_t GetCapacity() const { return static_cast(stackEnd_ - stack_); }
180 |
181 | private:
182 | template
183 | void Expand(size_t count) {
184 | // Only expand the capacity if the current stack exists. Otherwise just create a stack with initial capacity.
185 | size_t newCapacity;
186 | if (stack_ == 0) {
187 | if (!allocator_)
188 | ownAllocator_ = allocator_ = RAPIDJSON_NEW(Allocator)();
189 | newCapacity = initialCapacity_;
190 | } else {
191 | newCapacity = GetCapacity();
192 | newCapacity += (newCapacity + 1) / 2;
193 | }
194 | size_t newSize = GetSize() + sizeof(T) * count;
195 | if (newCapacity < newSize)
196 | newCapacity = newSize;
197 |
198 | Resize(newCapacity);
199 | }
200 |
201 | void Resize(size_t newCapacity) {
202 | const size_t size = GetSize(); // Backup the current size
203 | stack_ = static_cast(allocator_->Realloc(stack_, GetCapacity(), newCapacity));
204 | stackTop_ = stack_ + size;
205 | stackEnd_ = stack_ + newCapacity;
206 | }
207 |
208 | void Destroy() {
209 | Allocator::Free(stack_);
210 | RAPIDJSON_DELETE(ownAllocator_); // Only delete if it is owned by the stack
211 | }
212 |
213 | // Prohibit copy constructor & assignment operator.
214 | Stack(const Stack&);
215 | Stack& operator=(const Stack&);
216 |
217 | Allocator* allocator_;
218 | Allocator* ownAllocator_;
219 | char *stack_;
220 | char *stackTop_;
221 | char *stackEnd_;
222 | size_t initialCapacity_;
223 | };
224 |
225 | } // namespace internal
226 | RAPIDJSON_NAMESPACE_END
227 |
228 | #if defined(__clang__)
229 | RAPIDJSON_DIAG_POP
230 | #endif
231 |
232 | #endif // RAPIDJSON_STACK_H_
233 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/internal/strfunc.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_INTERNAL_STRFUNC_H_
16 | #define RAPIDJSON_INTERNAL_STRFUNC_H_
17 |
18 | #include "../stream.h"
19 | #include
20 |
21 | RAPIDJSON_NAMESPACE_BEGIN
22 | namespace internal {
23 |
24 | //! Custom strlen() which works on different character types.
25 | /*! \tparam Ch Character type (e.g. char, wchar_t, short)
26 | \param s Null-terminated input string.
27 | \return Number of characters in the string.
28 | \note This has the same semantics as strlen(), the return value is not number of Unicode codepoints.
29 | */
30 | template
31 | inline SizeType StrLen(const Ch* s) {
32 | RAPIDJSON_ASSERT(s != 0);
33 | const Ch* p = s;
34 | while (*p) ++p;
35 | return SizeType(p - s);
36 | }
37 |
38 | template <>
39 | inline SizeType StrLen(const char* s) {
40 | return SizeType(std::strlen(s));
41 | }
42 |
43 | template <>
44 | inline SizeType StrLen(const wchar_t* s) {
45 | return SizeType(std::wcslen(s));
46 | }
47 |
48 | //! Returns number of code points in a encoded string.
49 | template
50 | bool CountStringCodePoint(const typename Encoding::Ch* s, SizeType length, SizeType* outCount) {
51 | RAPIDJSON_ASSERT(s != 0);
52 | RAPIDJSON_ASSERT(outCount != 0);
53 | GenericStringStream is(s);
54 | const typename Encoding::Ch* end = s + length;
55 | SizeType count = 0;
56 | while (is.src_ < end) {
57 | unsigned codepoint;
58 | if (!Encoding::Decode(is, &codepoint))
59 | return false;
60 | count++;
61 | }
62 | *outCount = count;
63 | return true;
64 | }
65 |
66 | } // namespace internal
67 | RAPIDJSON_NAMESPACE_END
68 |
69 | #endif // RAPIDJSON_INTERNAL_STRFUNC_H_
70 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/internal/strtod.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_STRTOD_
16 | #define RAPIDJSON_STRTOD_
17 |
18 | #include "ieee754.h"
19 | #include "biginteger.h"
20 | #include "diyfp.h"
21 | #include "pow10.h"
22 | #include
23 | #include
24 |
25 | RAPIDJSON_NAMESPACE_BEGIN
26 | namespace internal {
27 |
28 | inline double FastPath(double significand, int exp) {
29 | if (exp < -308)
30 | return 0.0;
31 | else if (exp >= 0)
32 | return significand * internal::Pow10(exp);
33 | else
34 | return significand / internal::Pow10(-exp);
35 | }
36 |
37 | inline double StrtodNormalPrecision(double d, int p) {
38 | if (p < -308) {
39 | // Prevent expSum < -308, making Pow10(p) = 0
40 | d = FastPath(d, -308);
41 | d = FastPath(d, p + 308);
42 | }
43 | else
44 | d = FastPath(d, p);
45 | return d;
46 | }
47 |
48 | template
49 | inline T Min3(T a, T b, T c) {
50 | T m = a;
51 | if (m > b) m = b;
52 | if (m > c) m = c;
53 | return m;
54 | }
55 |
56 | inline int CheckWithinHalfULP(double b, const BigInteger& d, int dExp) {
57 | const Double db(b);
58 | const uint64_t bInt = db.IntegerSignificand();
59 | const int bExp = db.IntegerExponent();
60 | const int hExp = bExp - 1;
61 |
62 | int dS_Exp2 = 0, dS_Exp5 = 0, bS_Exp2 = 0, bS_Exp5 = 0, hS_Exp2 = 0, hS_Exp5 = 0;
63 |
64 | // Adjust for decimal exponent
65 | if (dExp >= 0) {
66 | dS_Exp2 += dExp;
67 | dS_Exp5 += dExp;
68 | }
69 | else {
70 | bS_Exp2 -= dExp;
71 | bS_Exp5 -= dExp;
72 | hS_Exp2 -= dExp;
73 | hS_Exp5 -= dExp;
74 | }
75 |
76 | // Adjust for binary exponent
77 | if (bExp >= 0)
78 | bS_Exp2 += bExp;
79 | else {
80 | dS_Exp2 -= bExp;
81 | hS_Exp2 -= bExp;
82 | }
83 |
84 | // Adjust for half ulp exponent
85 | if (hExp >= 0)
86 | hS_Exp2 += hExp;
87 | else {
88 | dS_Exp2 -= hExp;
89 | bS_Exp2 -= hExp;
90 | }
91 |
92 | // Remove common power of two factor from all three scaled values
93 | int common_Exp2 = Min3(dS_Exp2, bS_Exp2, hS_Exp2);
94 | dS_Exp2 -= common_Exp2;
95 | bS_Exp2 -= common_Exp2;
96 | hS_Exp2 -= common_Exp2;
97 |
98 | BigInteger dS = d;
99 | dS.MultiplyPow5(static_cast(dS_Exp5)) <<= static_cast(dS_Exp2);
100 |
101 | BigInteger bS(bInt);
102 | bS.MultiplyPow5(static_cast(bS_Exp5)) <<= static_cast(bS_Exp2);
103 |
104 | BigInteger hS(1);
105 | hS.MultiplyPow5(static_cast(hS_Exp5)) <<= static_cast(hS_Exp2);
106 |
107 | BigInteger delta(0);
108 | dS.Difference(bS, &delta);
109 |
110 | return delta.Compare(hS);
111 | }
112 |
113 | inline bool StrtodFast(double d, int p, double* result) {
114 | // Use fast path for string-to-double conversion if possible
115 | // see http://www.exploringbinary.com/fast-path-decimal-to-floating-point-conversion/
116 | if (p > 22 && p < 22 + 16) {
117 | // Fast Path Cases In Disguise
118 | d *= internal::Pow10(p - 22);
119 | p = 22;
120 | }
121 |
122 | if (p >= -22 && p <= 22 && d <= 9007199254740991.0) { // 2^53 - 1
123 | *result = FastPath(d, p);
124 | return true;
125 | }
126 | else
127 | return false;
128 | }
129 |
130 | // Compute an approximation and see if it is within 1/2 ULP
131 | inline bool StrtodDiyFp(const char* decimals, int dLen, int dExp, double* result) {
132 | uint64_t significand = 0;
133 | int i = 0; // 2^64 - 1 = 18446744073709551615, 1844674407370955161 = 0x1999999999999999
134 | for (; i < dLen; i++) {
135 | if (significand > RAPIDJSON_UINT64_C2(0x19999999, 0x99999999) ||
136 | (significand == RAPIDJSON_UINT64_C2(0x19999999, 0x99999999) && decimals[i] > '5'))
137 | break;
138 | significand = significand * 10u + static_cast(decimals[i] - '0');
139 | }
140 |
141 | if (i < dLen && decimals[i] >= '5') // Rounding
142 | significand++;
143 |
144 | int remaining = dLen - i;
145 | const int kUlpShift = 3;
146 | const int kUlp = 1 << kUlpShift;
147 | int64_t error = (remaining == 0) ? 0 : kUlp / 2;
148 |
149 | DiyFp v(significand, 0);
150 | v = v.Normalize();
151 | error <<= -v.e;
152 |
153 | dExp += remaining;
154 |
155 | int actualExp;
156 | DiyFp cachedPower = GetCachedPower10(dExp, &actualExp);
157 | if (actualExp != dExp) {
158 | static const DiyFp kPow10[] = {
159 | DiyFp(RAPIDJSON_UINT64_C2(0xa0000000, 0x00000000), -60), // 10^1
160 | DiyFp(RAPIDJSON_UINT64_C2(0xc8000000, 0x00000000), -57), // 10^2
161 | DiyFp(RAPIDJSON_UINT64_C2(0xfa000000, 0x00000000), -54), // 10^3
162 | DiyFp(RAPIDJSON_UINT64_C2(0x9c400000, 0x00000000), -50), // 10^4
163 | DiyFp(RAPIDJSON_UINT64_C2(0xc3500000, 0x00000000), -47), // 10^5
164 | DiyFp(RAPIDJSON_UINT64_C2(0xf4240000, 0x00000000), -44), // 10^6
165 | DiyFp(RAPIDJSON_UINT64_C2(0x98968000, 0x00000000), -40) // 10^7
166 | };
167 | int adjustment = dExp - actualExp;
168 | RAPIDJSON_ASSERT(adjustment >= 1 && adjustment < 8);
169 | v = v * kPow10[adjustment - 1];
170 | if (dLen + adjustment > 19) // has more digits than decimal digits in 64-bit
171 | error += kUlp / 2;
172 | }
173 |
174 | v = v * cachedPower;
175 |
176 | error += kUlp + (error == 0 ? 0 : 1);
177 |
178 | const int oldExp = v.e;
179 | v = v.Normalize();
180 | error <<= oldExp - v.e;
181 |
182 | const int effectiveSignificandSize = Double::EffectiveSignificandSize(64 + v.e);
183 | int precisionSize = 64 - effectiveSignificandSize;
184 | if (precisionSize + kUlpShift >= 64) {
185 | int scaleExp = (precisionSize + kUlpShift) - 63;
186 | v.f >>= scaleExp;
187 | v.e += scaleExp;
188 | error = (error >> scaleExp) + 1 + kUlp;
189 | precisionSize -= scaleExp;
190 | }
191 |
192 | DiyFp rounded(v.f >> precisionSize, v.e + precisionSize);
193 | const uint64_t precisionBits = (v.f & ((uint64_t(1) << precisionSize) - 1)) * kUlp;
194 | const uint64_t halfWay = (uint64_t(1) << (precisionSize - 1)) * kUlp;
195 | if (precisionBits >= halfWay + static_cast(error)) {
196 | rounded.f++;
197 | if (rounded.f & (DiyFp::kDpHiddenBit << 1)) { // rounding overflows mantissa (issue #340)
198 | rounded.f >>= 1;
199 | rounded.e++;
200 | }
201 | }
202 |
203 | *result = rounded.ToDouble();
204 |
205 | return halfWay - static_cast(error) >= precisionBits || precisionBits >= halfWay + static_cast(error);
206 | }
207 |
208 | inline double StrtodBigInteger(double approx, const char* decimals, int dLen, int dExp) {
209 | RAPIDJSON_ASSERT(dLen >= 0);
210 | const BigInteger dInt(decimals, static_cast(dLen));
211 | Double a(approx);
212 | int cmp = CheckWithinHalfULP(a.Value(), dInt, dExp);
213 | if (cmp < 0)
214 | return a.Value(); // within half ULP
215 | else if (cmp == 0) {
216 | // Round towards even
217 | if (a.Significand() & 1)
218 | return a.NextPositiveDouble();
219 | else
220 | return a.Value();
221 | }
222 | else // adjustment
223 | return a.NextPositiveDouble();
224 | }
225 |
226 | inline double StrtodFullPrecision(double d, int p, const char* decimals, size_t length, size_t decimalPosition, int exp) {
227 | RAPIDJSON_ASSERT(d >= 0.0);
228 | RAPIDJSON_ASSERT(length >= 1);
229 |
230 | double result = 0.0;
231 | if (StrtodFast(d, p, &result))
232 | return result;
233 |
234 | RAPIDJSON_ASSERT(length <= INT_MAX);
235 | int dLen = static_cast(length);
236 |
237 | RAPIDJSON_ASSERT(length >= decimalPosition);
238 | RAPIDJSON_ASSERT(length - decimalPosition <= INT_MAX);
239 | int dExpAdjust = static_cast(length - decimalPosition);
240 |
241 | RAPIDJSON_ASSERT(exp >= INT_MIN + dExpAdjust);
242 | int dExp = exp - dExpAdjust;
243 |
244 | // Make sure length+dExp does not overflow
245 | RAPIDJSON_ASSERT(dExp <= INT_MAX - dLen);
246 |
247 | // Trim leading zeros
248 | while (dLen > 0 && *decimals == '0') {
249 | dLen--;
250 | decimals++;
251 | }
252 |
253 | // Trim trailing zeros
254 | while (dLen > 0 && decimals[dLen - 1] == '0') {
255 | dLen--;
256 | dExp++;
257 | }
258 |
259 | if (dLen == 0) { // Buffer only contains zeros.
260 | return 0.0;
261 | }
262 |
263 | // Trim right-most digits
264 | const int kMaxDecimalDigit = 767 + 1;
265 | if (dLen > kMaxDecimalDigit) {
266 | dExp += dLen - kMaxDecimalDigit;
267 | dLen = kMaxDecimalDigit;
268 | }
269 |
270 | // If too small, underflow to zero.
271 | // Any x <= 10^-324 is interpreted as zero.
272 | if (dLen + dExp <= -324)
273 | return 0.0;
274 |
275 | // If too large, overflow to infinity.
276 | // Any x >= 10^309 is interpreted as +infinity.
277 | if (dLen + dExp > 309)
278 | return std::numeric_limits::infinity();
279 |
280 | if (StrtodDiyFp(decimals, dLen, dExp, &result))
281 | return result;
282 |
283 | // Use approximation from StrtodDiyFp and make adjustment with BigInteger comparison
284 | return StrtodBigInteger(result, decimals, dLen, dExp);
285 | }
286 |
287 | } // namespace internal
288 | RAPIDJSON_NAMESPACE_END
289 |
290 | #endif // RAPIDJSON_STRTOD_
291 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/internal/swap.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_INTERNAL_SWAP_H_
16 | #define RAPIDJSON_INTERNAL_SWAP_H_
17 |
18 | #include "../rapidjson.h"
19 |
20 | #if defined(__clang__)
21 | RAPIDJSON_DIAG_PUSH
22 | RAPIDJSON_DIAG_OFF(c++98-compat)
23 | #endif
24 |
25 | RAPIDJSON_NAMESPACE_BEGIN
26 | namespace internal {
27 |
28 | //! Custom swap() to avoid dependency on C++ header
29 | /*! \tparam T Type of the arguments to swap, should be instantiated with primitive C++ types only.
30 | \note This has the same semantics as std::swap().
31 | */
32 | template
33 | inline void Swap(T& a, T& b) RAPIDJSON_NOEXCEPT {
34 | T tmp = a;
35 | a = b;
36 | b = tmp;
37 | }
38 |
39 | } // namespace internal
40 | RAPIDJSON_NAMESPACE_END
41 |
42 | #if defined(__clang__)
43 | RAPIDJSON_DIAG_POP
44 | #endif
45 |
46 | #endif // RAPIDJSON_INTERNAL_SWAP_H_
47 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/istreamwrapper.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_ISTREAMWRAPPER_H_
16 | #define RAPIDJSON_ISTREAMWRAPPER_H_
17 |
18 | #include "stream.h"
19 | #include
20 | #include
21 |
22 | #ifdef __clang__
23 | RAPIDJSON_DIAG_PUSH
24 | RAPIDJSON_DIAG_OFF(padded)
25 | #elif defined(_MSC_VER)
26 | RAPIDJSON_DIAG_PUSH
27 | RAPIDJSON_DIAG_OFF(4351) // new behavior: elements of array 'array' will be default initialized
28 | #endif
29 |
30 | RAPIDJSON_NAMESPACE_BEGIN
31 |
32 | //! Wrapper of \c std::basic_istream into RapidJSON's Stream concept.
33 | /*!
34 | The classes can be wrapped including but not limited to:
35 |
36 | - \c std::istringstream
37 | - \c std::stringstream
38 | - \c std::wistringstream
39 | - \c std::wstringstream
40 | - \c std::ifstream
41 | - \c std::fstream
42 | - \c std::wifstream
43 | - \c std::wfstream
44 |
45 | \tparam StreamType Class derived from \c std::basic_istream.
46 | */
47 |
48 | template
49 | class BasicIStreamWrapper {
50 | public:
51 | typedef typename StreamType::char_type Ch;
52 |
53 | //! Constructor.
54 | /*!
55 | \param stream stream opened for read.
56 | */
57 | BasicIStreamWrapper(StreamType &stream) : stream_(stream), buffer_(peekBuffer_), bufferSize_(4), bufferLast_(0), current_(buffer_), readCount_(0), count_(0), eof_(false) {
58 | Read();
59 | }
60 |
61 | //! Constructor.
62 | /*!
63 | \param stream stream opened for read.
64 | \param buffer user-supplied buffer.
65 | \param bufferSize size of buffer in bytes. Must >=4 bytes.
66 | */
67 | BasicIStreamWrapper(StreamType &stream, char* buffer, size_t bufferSize) : stream_(stream), buffer_(buffer), bufferSize_(bufferSize), bufferLast_(0), current_(buffer_), readCount_(0), count_(0), eof_(false) {
68 | RAPIDJSON_ASSERT(bufferSize >= 4);
69 | Read();
70 | }
71 |
72 | Ch Peek() const { return *current_; }
73 | Ch Take() { Ch c = *current_; Read(); return c; }
74 | size_t Tell() const { return count_ + static_cast(current_ - buffer_); }
75 |
76 | // Not implemented
77 | void Put(Ch) { RAPIDJSON_ASSERT(false); }
78 | void Flush() { RAPIDJSON_ASSERT(false); }
79 | Ch* PutBegin() { RAPIDJSON_ASSERT(false); return 0; }
80 | size_t PutEnd(Ch*) { RAPIDJSON_ASSERT(false); return 0; }
81 |
82 | // For encoding detection only.
83 | const Ch* Peek4() const {
84 | return (current_ + 4 - !eof_ <= bufferLast_) ? current_ : 0;
85 | }
86 |
87 | private:
88 | BasicIStreamWrapper();
89 | BasicIStreamWrapper(const BasicIStreamWrapper&);
90 | BasicIStreamWrapper& operator=(const BasicIStreamWrapper&);
91 |
92 | void Read() {
93 | if (current_ < bufferLast_)
94 | ++current_;
95 | else if (!eof_) {
96 | count_ += readCount_;
97 | readCount_ = bufferSize_;
98 | bufferLast_ = buffer_ + readCount_ - 1;
99 | current_ = buffer_;
100 |
101 | if (!stream_.read(buffer_, static_cast(bufferSize_))) {
102 | readCount_ = static_cast(stream_.gcount());
103 | *(bufferLast_ = buffer_ + readCount_) = '\0';
104 | eof_ = true;
105 | }
106 | }
107 | }
108 |
109 | StreamType &stream_;
110 | Ch peekBuffer_[4], *buffer_;
111 | size_t bufferSize_;
112 | Ch *bufferLast_;
113 | Ch *current_;
114 | size_t readCount_;
115 | size_t count_; //!< Number of characters read
116 | bool eof_;
117 | };
118 |
119 | typedef BasicIStreamWrapper IStreamWrapper;
120 | typedef BasicIStreamWrapper WIStreamWrapper;
121 |
122 | #if defined(__clang__) || defined(_MSC_VER)
123 | RAPIDJSON_DIAG_POP
124 | #endif
125 |
126 | RAPIDJSON_NAMESPACE_END
127 |
128 | #endif // RAPIDJSON_ISTREAMWRAPPER_H_
129 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/memorybuffer.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_MEMORYBUFFER_H_
16 | #define RAPIDJSON_MEMORYBUFFER_H_
17 |
18 | #include "stream.h"
19 | #include "internal/stack.h"
20 |
21 | RAPIDJSON_NAMESPACE_BEGIN
22 |
23 | //! Represents an in-memory output byte stream.
24 | /*!
25 | This class is mainly for being wrapped by EncodedOutputStream or AutoUTFOutputStream.
26 |
27 | It is similar to FileWriteBuffer but the destination is an in-memory buffer instead of a file.
28 |
29 | Differences between MemoryBuffer and StringBuffer:
30 | 1. StringBuffer has Encoding but MemoryBuffer is only a byte buffer.
31 | 2. StringBuffer::GetString() returns a null-terminated string. MemoryBuffer::GetBuffer() returns a buffer without terminator.
32 |
33 | \tparam Allocator type for allocating memory buffer.
34 | \note implements Stream concept
35 | */
36 | template
37 | struct GenericMemoryBuffer {
38 | typedef char Ch; // byte
39 |
40 | GenericMemoryBuffer(Allocator* allocator = 0, size_t capacity = kDefaultCapacity) : stack_(allocator, capacity) {}
41 |
42 | void Put(Ch c) { *stack_.template Push() = c; }
43 | void Flush() {}
44 |
45 | void Clear() { stack_.Clear(); }
46 | void ShrinkToFit() { stack_.ShrinkToFit(); }
47 | Ch* Push(size_t count) { return stack_.template Push(count); }
48 | void Pop(size_t count) { stack_.template Pop(count); }
49 |
50 | const Ch* GetBuffer() const {
51 | return stack_.template Bottom();
52 | }
53 |
54 | size_t GetSize() const { return stack_.GetSize(); }
55 |
56 | static const size_t kDefaultCapacity = 256;
57 | mutable internal::Stack stack_;
58 | };
59 |
60 | typedef GenericMemoryBuffer<> MemoryBuffer;
61 |
62 | //! Implement specialized version of PutN() with memset() for better performance.
63 | template<>
64 | inline void PutN(MemoryBuffer& memoryBuffer, char c, size_t n) {
65 | std::memset(memoryBuffer.stack_.Push(n), c, n * sizeof(c));
66 | }
67 |
68 | RAPIDJSON_NAMESPACE_END
69 |
70 | #endif // RAPIDJSON_MEMORYBUFFER_H_
71 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/memorystream.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_MEMORYSTREAM_H_
16 | #define RAPIDJSON_MEMORYSTREAM_H_
17 |
18 | #include "stream.h"
19 |
20 | #ifdef __clang__
21 | RAPIDJSON_DIAG_PUSH
22 | RAPIDJSON_DIAG_OFF(unreachable-code)
23 | RAPIDJSON_DIAG_OFF(missing-noreturn)
24 | #endif
25 |
26 | RAPIDJSON_NAMESPACE_BEGIN
27 |
28 | //! Represents an in-memory input byte stream.
29 | /*!
30 | This class is mainly for being wrapped by EncodedInputStream or AutoUTFInputStream.
31 |
32 | It is similar to FileReadBuffer but the source is an in-memory buffer instead of a file.
33 |
34 | Differences between MemoryStream and StringStream:
35 | 1. StringStream has encoding but MemoryStream is a byte stream.
36 | 2. MemoryStream needs size of the source buffer and the buffer don't need to be null terminated. StringStream assume null-terminated string as source.
37 | 3. MemoryStream supports Peek4() for encoding detection. StringStream is specified with an encoding so it should not have Peek4().
38 | \note implements Stream concept
39 | */
40 | struct MemoryStream {
41 | typedef char Ch; // byte
42 |
43 | MemoryStream(const Ch *src, size_t size) : src_(src), begin_(src), end_(src + size), size_(size) {}
44 |
45 | Ch Peek() const { return RAPIDJSON_UNLIKELY(src_ == end_) ? '\0' : *src_; }
46 | Ch Take() { return RAPIDJSON_UNLIKELY(src_ == end_) ? '\0' : *src_++; }
47 | size_t Tell() const { return static_cast(src_ - begin_); }
48 |
49 | Ch* PutBegin() { RAPIDJSON_ASSERT(false); return 0; }
50 | void Put(Ch) { RAPIDJSON_ASSERT(false); }
51 | void Flush() { RAPIDJSON_ASSERT(false); }
52 | size_t PutEnd(Ch*) { RAPIDJSON_ASSERT(false); return 0; }
53 |
54 | // For encoding detection only.
55 | const Ch* Peek4() const {
56 | return Tell() + 4 <= size_ ? src_ : 0;
57 | }
58 |
59 | const Ch* src_; //!< Current read position.
60 | const Ch* begin_; //!< Original head of the string.
61 | const Ch* end_; //!< End of stream.
62 | size_t size_; //!< Size of the stream.
63 | };
64 |
65 | RAPIDJSON_NAMESPACE_END
66 |
67 | #ifdef __clang__
68 | RAPIDJSON_DIAG_POP
69 | #endif
70 |
71 | #endif // RAPIDJSON_MEMORYBUFFER_H_
72 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/msinttypes/inttypes.h:
--------------------------------------------------------------------------------
1 | // ISO C9x compliant inttypes.h for Microsoft Visual Studio
2 | // Based on ISO/IEC 9899:TC2 Committee draft (May 6, 2005) WG14/N1124
3 | //
4 | // Copyright (c) 2006-2013 Alexander Chemeris
5 | //
6 | // Redistribution and use in source and binary forms, with or without
7 | // modification, are permitted provided that the following conditions are met:
8 | //
9 | // 1. Redistributions of source code must retain the above copyright notice,
10 | // this list of conditions and the following disclaimer.
11 | //
12 | // 2. Redistributions in binary form must reproduce the above copyright
13 | // notice, this list of conditions and the following disclaimer in the
14 | // documentation and/or other materials provided with the distribution.
15 | //
16 | // 3. Neither the name of the product nor the names of its contributors may
17 | // be used to endorse or promote products derived from this software
18 | // without specific prior written permission.
19 | //
20 | // THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
21 | // WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
22 | // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
23 | // EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 | // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
25 | // PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
26 | // OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
27 | // WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
28 | // OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
29 | // ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30 | //
31 | ///////////////////////////////////////////////////////////////////////////////
32 |
33 | // The above software in this distribution may have been modified by
34 | // THL A29 Limited ("Tencent Modifications").
35 | // All Tencent Modifications are Copyright (C) 2015 THL A29 Limited.
36 |
37 | #ifndef _MSC_VER // [
38 | #error "Use this header only with Microsoft Visual C++ compilers!"
39 | #endif // _MSC_VER ]
40 |
41 | #ifndef _MSC_INTTYPES_H_ // [
42 | #define _MSC_INTTYPES_H_
43 |
44 | #if _MSC_VER > 1000
45 | #pragma once
46 | #endif
47 |
48 | #include "stdint.h"
49 |
50 | // miloyip: VC supports inttypes.h since VC2013
51 | #if _MSC_VER >= 1800
52 | #include
53 | #else
54 |
55 | // 7.8 Format conversion of integer types
56 |
57 | typedef struct {
58 | intmax_t quot;
59 | intmax_t rem;
60 | } imaxdiv_t;
61 |
62 | // 7.8.1 Macros for format specifiers
63 |
64 | #if !defined(__cplusplus) || defined(__STDC_FORMAT_MACROS) // [ See footnote 185 at page 198
65 |
66 | // The fprintf macros for signed integers are:
67 | #define PRId8 "d"
68 | #define PRIi8 "i"
69 | #define PRIdLEAST8 "d"
70 | #define PRIiLEAST8 "i"
71 | #define PRIdFAST8 "d"
72 | #define PRIiFAST8 "i"
73 |
74 | #define PRId16 "hd"
75 | #define PRIi16 "hi"
76 | #define PRIdLEAST16 "hd"
77 | #define PRIiLEAST16 "hi"
78 | #define PRIdFAST16 "hd"
79 | #define PRIiFAST16 "hi"
80 |
81 | #define PRId32 "I32d"
82 | #define PRIi32 "I32i"
83 | #define PRIdLEAST32 "I32d"
84 | #define PRIiLEAST32 "I32i"
85 | #define PRIdFAST32 "I32d"
86 | #define PRIiFAST32 "I32i"
87 |
88 | #define PRId64 "I64d"
89 | #define PRIi64 "I64i"
90 | #define PRIdLEAST64 "I64d"
91 | #define PRIiLEAST64 "I64i"
92 | #define PRIdFAST64 "I64d"
93 | #define PRIiFAST64 "I64i"
94 |
95 | #define PRIdMAX "I64d"
96 | #define PRIiMAX "I64i"
97 |
98 | #define PRIdPTR "Id"
99 | #define PRIiPTR "Ii"
100 |
101 | // The fprintf macros for unsigned integers are:
102 | #define PRIo8 "o"
103 | #define PRIu8 "u"
104 | #define PRIx8 "x"
105 | #define PRIX8 "X"
106 | #define PRIoLEAST8 "o"
107 | #define PRIuLEAST8 "u"
108 | #define PRIxLEAST8 "x"
109 | #define PRIXLEAST8 "X"
110 | #define PRIoFAST8 "o"
111 | #define PRIuFAST8 "u"
112 | #define PRIxFAST8 "x"
113 | #define PRIXFAST8 "X"
114 |
115 | #define PRIo16 "ho"
116 | #define PRIu16 "hu"
117 | #define PRIx16 "hx"
118 | #define PRIX16 "hX"
119 | #define PRIoLEAST16 "ho"
120 | #define PRIuLEAST16 "hu"
121 | #define PRIxLEAST16 "hx"
122 | #define PRIXLEAST16 "hX"
123 | #define PRIoFAST16 "ho"
124 | #define PRIuFAST16 "hu"
125 | #define PRIxFAST16 "hx"
126 | #define PRIXFAST16 "hX"
127 |
128 | #define PRIo32 "I32o"
129 | #define PRIu32 "I32u"
130 | #define PRIx32 "I32x"
131 | #define PRIX32 "I32X"
132 | #define PRIoLEAST32 "I32o"
133 | #define PRIuLEAST32 "I32u"
134 | #define PRIxLEAST32 "I32x"
135 | #define PRIXLEAST32 "I32X"
136 | #define PRIoFAST32 "I32o"
137 | #define PRIuFAST32 "I32u"
138 | #define PRIxFAST32 "I32x"
139 | #define PRIXFAST32 "I32X"
140 |
141 | #define PRIo64 "I64o"
142 | #define PRIu64 "I64u"
143 | #define PRIx64 "I64x"
144 | #define PRIX64 "I64X"
145 | #define PRIoLEAST64 "I64o"
146 | #define PRIuLEAST64 "I64u"
147 | #define PRIxLEAST64 "I64x"
148 | #define PRIXLEAST64 "I64X"
149 | #define PRIoFAST64 "I64o"
150 | #define PRIuFAST64 "I64u"
151 | #define PRIxFAST64 "I64x"
152 | #define PRIXFAST64 "I64X"
153 |
154 | #define PRIoMAX "I64o"
155 | #define PRIuMAX "I64u"
156 | #define PRIxMAX "I64x"
157 | #define PRIXMAX "I64X"
158 |
159 | #define PRIoPTR "Io"
160 | #define PRIuPTR "Iu"
161 | #define PRIxPTR "Ix"
162 | #define PRIXPTR "IX"
163 |
164 | // The fscanf macros for signed integers are:
165 | #define SCNd8 "d"
166 | #define SCNi8 "i"
167 | #define SCNdLEAST8 "d"
168 | #define SCNiLEAST8 "i"
169 | #define SCNdFAST8 "d"
170 | #define SCNiFAST8 "i"
171 |
172 | #define SCNd16 "hd"
173 | #define SCNi16 "hi"
174 | #define SCNdLEAST16 "hd"
175 | #define SCNiLEAST16 "hi"
176 | #define SCNdFAST16 "hd"
177 | #define SCNiFAST16 "hi"
178 |
179 | #define SCNd32 "ld"
180 | #define SCNi32 "li"
181 | #define SCNdLEAST32 "ld"
182 | #define SCNiLEAST32 "li"
183 | #define SCNdFAST32 "ld"
184 | #define SCNiFAST32 "li"
185 |
186 | #define SCNd64 "I64d"
187 | #define SCNi64 "I64i"
188 | #define SCNdLEAST64 "I64d"
189 | #define SCNiLEAST64 "I64i"
190 | #define SCNdFAST64 "I64d"
191 | #define SCNiFAST64 "I64i"
192 |
193 | #define SCNdMAX "I64d"
194 | #define SCNiMAX "I64i"
195 |
196 | #ifdef _WIN64 // [
197 | # define SCNdPTR "I64d"
198 | # define SCNiPTR "I64i"
199 | #else // _WIN64 ][
200 | # define SCNdPTR "ld"
201 | # define SCNiPTR "li"
202 | #endif // _WIN64 ]
203 |
204 | // The fscanf macros for unsigned integers are:
205 | #define SCNo8 "o"
206 | #define SCNu8 "u"
207 | #define SCNx8 "x"
208 | #define SCNX8 "X"
209 | #define SCNoLEAST8 "o"
210 | #define SCNuLEAST8 "u"
211 | #define SCNxLEAST8 "x"
212 | #define SCNXLEAST8 "X"
213 | #define SCNoFAST8 "o"
214 | #define SCNuFAST8 "u"
215 | #define SCNxFAST8 "x"
216 | #define SCNXFAST8 "X"
217 |
218 | #define SCNo16 "ho"
219 | #define SCNu16 "hu"
220 | #define SCNx16 "hx"
221 | #define SCNX16 "hX"
222 | #define SCNoLEAST16 "ho"
223 | #define SCNuLEAST16 "hu"
224 | #define SCNxLEAST16 "hx"
225 | #define SCNXLEAST16 "hX"
226 | #define SCNoFAST16 "ho"
227 | #define SCNuFAST16 "hu"
228 | #define SCNxFAST16 "hx"
229 | #define SCNXFAST16 "hX"
230 |
231 | #define SCNo32 "lo"
232 | #define SCNu32 "lu"
233 | #define SCNx32 "lx"
234 | #define SCNX32 "lX"
235 | #define SCNoLEAST32 "lo"
236 | #define SCNuLEAST32 "lu"
237 | #define SCNxLEAST32 "lx"
238 | #define SCNXLEAST32 "lX"
239 | #define SCNoFAST32 "lo"
240 | #define SCNuFAST32 "lu"
241 | #define SCNxFAST32 "lx"
242 | #define SCNXFAST32 "lX"
243 |
244 | #define SCNo64 "I64o"
245 | #define SCNu64 "I64u"
246 | #define SCNx64 "I64x"
247 | #define SCNX64 "I64X"
248 | #define SCNoLEAST64 "I64o"
249 | #define SCNuLEAST64 "I64u"
250 | #define SCNxLEAST64 "I64x"
251 | #define SCNXLEAST64 "I64X"
252 | #define SCNoFAST64 "I64o"
253 | #define SCNuFAST64 "I64u"
254 | #define SCNxFAST64 "I64x"
255 | #define SCNXFAST64 "I64X"
256 |
257 | #define SCNoMAX "I64o"
258 | #define SCNuMAX "I64u"
259 | #define SCNxMAX "I64x"
260 | #define SCNXMAX "I64X"
261 |
262 | #ifdef _WIN64 // [
263 | # define SCNoPTR "I64o"
264 | # define SCNuPTR "I64u"
265 | # define SCNxPTR "I64x"
266 | # define SCNXPTR "I64X"
267 | #else // _WIN64 ][
268 | # define SCNoPTR "lo"
269 | # define SCNuPTR "lu"
270 | # define SCNxPTR "lx"
271 | # define SCNXPTR "lX"
272 | #endif // _WIN64 ]
273 |
274 | #endif // __STDC_FORMAT_MACROS ]
275 |
276 | // 7.8.2 Functions for greatest-width integer types
277 |
278 | // 7.8.2.1 The imaxabs function
279 | #define imaxabs _abs64
280 |
281 | // 7.8.2.2 The imaxdiv function
282 |
283 | // This is modified version of div() function from Microsoft's div.c found
284 | // in %MSVC.NET%\crt\src\div.c
285 | #ifdef STATIC_IMAXDIV // [
286 | static
287 | #else // STATIC_IMAXDIV ][
288 | _inline
289 | #endif // STATIC_IMAXDIV ]
290 | imaxdiv_t __cdecl imaxdiv(intmax_t numer, intmax_t denom)
291 | {
292 | imaxdiv_t result;
293 |
294 | result.quot = numer / denom;
295 | result.rem = numer % denom;
296 |
297 | if (numer < 0 && result.rem > 0) {
298 | // did division wrong; must fix up
299 | ++result.quot;
300 | result.rem -= denom;
301 | }
302 |
303 | return result;
304 | }
305 |
306 | // 7.8.2.3 The strtoimax and strtoumax functions
307 | #define strtoimax _strtoi64
308 | #define strtoumax _strtoui64
309 |
310 | // 7.8.2.4 The wcstoimax and wcstoumax functions
311 | #define wcstoimax _wcstoi64
312 | #define wcstoumax _wcstoui64
313 |
314 | #endif // _MSC_VER >= 1800
315 |
316 | #endif // _MSC_INTTYPES_H_ ]
317 |
--------------------------------------------------------------------------------
/UnityFPSUnlocker/third/rapidjson/rapidjson/ostreamwrapper.h:
--------------------------------------------------------------------------------
1 | // Tencent is pleased to support the open source community by making RapidJSON available.
2 | //
3 | // Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved.
4 | //
5 | // Licensed under the MIT License (the "License"); you may not use this file except
6 | // in compliance with the License. You may obtain a copy of the License at
7 | //
8 | // http://opensource.org/licenses/MIT
9 | //
10 | // Unless required by applicable law or agreed to in writing, software distributed
11 | // under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
12 | // CONDITIONS OF ANY KIND, either express or implied. See the License for the
13 | // specific language governing permissions and limitations under the License.
14 |
15 | #ifndef RAPIDJSON_OSTREAMWRAPPER_H_
16 | #define RAPIDJSON_OSTREAMWRAPPER_H_
17 |
18 | #include "stream.h"
19 | #include
20 |
21 | #ifdef __clang__
22 | RAPIDJSON_DIAG_PUSH
23 | RAPIDJSON_DIAG_OFF(padded)
24 | #endif
25 |
26 | RAPIDJSON_NAMESPACE_BEGIN
27 |
28 | //! Wrapper of \c std::basic_ostream into RapidJSON's Stream concept.
29 | /*!
30 | The classes can be wrapped including but not limited to:
31 |
32 | - \c std::ostringstream
33 | - \c std::stringstream
34 | - \c std::wpstringstream
35 | - \c std::wstringstream
36 | - \c std::ifstream
37 | - \c std::fstream
38 | - \c std::wofstream
39 | - \c std::wfstream
40 |
41 | \tparam StreamType Class derived from \c std::basic_ostream.
42 | */
43 |
44 | template