Skip to content

Releases: MolotovCherry/Yet-Another-BG3-Native-Mod-Loader

v0.3.0

14 Jan 17:54
cfbe4b3
Compare
Choose a tag to compare

Highlights

Added autostart version that behaves like NML in that it automatically patches without requiring manual starting/stopping. This requires installation. (see INSTRUCTIONS.txt for how to use).

Notes

This comes with a new file loader.dll. No special setup is required. Just keep loader.dll in the same folder as the 2 exe's and everything will work.

Changes

Did a massive refactoring of most of the codebase. Plugin loading logic was separated out into a separate loader dll for flexibility.

  • Faster plugin loading. Once the loader is injected, it does the work of the actual plugin loading at full speed from within the actual process.
  • Also multi-threaded plugin loading to load plugins in parallel. Before this, they loaded sequentially.
  • The loader now calls any exported Init fn so that any plugin developers can properly program their plugin.1 (P.S. as a plugin dev, your init function can take as long as you want. But you should spawn your own thread so Init can finish in a timely manner)
  • Additionally to the above, developer plugins can also detect when yabg3ml is running vs other mod loaders. This allows you to do things like not run Init code in DllMain, and instead let my mod loader call Init. Otherwise, if it's not mine, you can Init from DllMain to keep compatibility with other mod loaders. See the plugin template for how to do this.
  • Note to plugin devs / mod loader makers: If you were using an old commit of native plugin lib, please update asap. There has been dev work on it and the previous metadata macro is incompatible with this version.
  • native plugin template for plugin devs has been updated and improved. Check it out if you used a previous version.
  • More robust dirty module check, which is accurate cause it checks windows file identifier.
  • Improved process watching code.
  • Use default config values if config is malformed. This will allow it to load if entries are missing, though note if you're doing default values it may load the wrong one since you haven't configured it. Also added popup if loading config failed so it's clearer what happened.
  • Grant SE_DEBUG_NAME if running as admin just to ensure we get extra permissions. This will ensure the program succeeds if it didn't work in non-admin mode
  • General stability improvements due to the refactor and testing.
  • Various ux improvements to not crash but rather display a popup with better information
  • Updated visual style of popups to look more modern.
  • Added log level to config file to allow easily adjusting the level.
  • Added warn popup to loader.dll if it was not loaded from support application.
  • If api call in patch check failed, sleep a little bit to let process catch up before retrying again. If it's still failing after a set timeout period, it's considered an error. (Error in the sense that process injection is aborted)
  • Greatly improved all the warning messages to clearly state what's happening, and that you can press OK and the app continues normally.
  • Refactored multithreaded communication and process shutdown system
  • Added tray icon to injector as well
  • Added tooltips to tray icon to see which tool is running
  • Made our own custom win32 event loop and removed winit dependency
  • Slightly decreased binary size by using a lighter cli arg parsing library (also removed the help and version flags since I doubt they're needed anyways; --cli will still spawn a cli window as usual)
  • Fixed long-time bug with the injector where after starting it, the mouse showed a busy loading icon for a long time even though the process worked normally. This was because we didn't run the event loop on this tool - now we do
  • Lots of general misc fixes and improvements

Default config file

Your config file may be missing some optional entries. Missing entries are filled with default values. For reference, here's what a full default config file looks like

[core]
enabled = true # whether plugin loading is enabled or not
install_root = 'C:\Program Files (x86)\Steam\steamapps\common\Baldurs Gate 3'
disabled_plugins = [] # list of plugins to disable
cli = false # show console window

[log]
level = "info" # increase verbosity of logs. useful when debugging. please set to "trace" if you have a bug to report
target = false

Full Changelog: v0.2.6...v0.3.0


  1. Why? Because actually doing anything non-basic inside DllMain is a very bad idea. Deadlocks, UB (even silent UB), and a whole host of other nasty things can happen if you use DllMain for anything except simple tasks. An Init function is the proper way to do it, so now we support it! This is also reflected in some updates to the plugin template. Also, if you're a dev, the Rust signature is extern "C" fn Init(); if you use C, it is void __cdecl Init(void). See articles below.
    https://devblogs.microsoft.com/oldnewthing/20070904-00/?p=25283
    https://devblogs.microsoft.com/oldnewthing/20040128-00/?p=40853
    https://devblogs.microsoft.com/oldnewthing/20040127-00/?p=40873
    https://devblogs.microsoft.com/oldnewthing/20100115-00/?p=15253
    https://blog.barthe.ph/2009/07/30/no-stdlib-in-dllmai/
    https://learn.microsoft.com/en-us/windows/win32/dlls/dllmain?redirectedfrom=MSDN (see warning section)
    https://learn.microsoft.com/en-us/windows/win32/dlls/dynamic-link-library-best-practices

v0.3.0 - Rc 2

11 Nov 21:14
26c968a
Compare
Choose a tag to compare
v0.3.0 - Rc 2 Pre-release
Pre-release

Changes since Rc 1

(also see previous rc 1, beta 2, and beta 1 for full 0.3.0 changelog)

  • Added autostart version that behaves like NML in that it automatically patches without requiring manual starting/stopping. This one requires installation, unlike the watcher and injector. (see INSTRUCTIONS.txt for how to use).

Full Changelog: v0.2.6...v0.3.0rc2
Full Changelog 0.2.6-b2: v0.2.6...v0.3.0b2
Full Changelog b1-b2: v0.3.0b1...v0.3.0b2
Full Changelog b2-rc1: v0.3.0b2...v0.3.0rc1
Full Changelog rc1-rc2: v0.3.0rc1...v0.3.0rc2

v0.3.0 - Rc 1

21 Oct 18:51
6896a27
Compare
Choose a tag to compare
v0.3.0 - Rc 1 Pre-release
Pre-release

If there are no reported problems with this version, this is probably the last rc before a full release.

Changes since Beta 2

(also see previous beta 2 and beta 1 for full 0.3.0 changelog)

  • Refactored multithreaded communication and process shutdown system
  • Added tray icon to injector as well
  • Added tooltips to tray icon to see which tool is running
  • Made our own custom win32 event loop and removed winit dependency
  • Slightly decreased binary size by using a lighter cli arg parsing library (also removed the help and version flags since I doubt they're needed anyways; --cli will still spawn a cli window as usual)
  • Fixed long-time bug with the injector where after starting it, the mouse showed a busy loading icon for a long time even though the process worked normally. This was because we didn't run the event loop on this tool - now we do

Full Changelog 0.2.6-b2: v0.2.6...v0.3.0b2
Full Changelog b1-b2: v0.3.0b1...v0.3.0b2
Full Changelog b2-rc1: v0.3.0b2...v0.3.0rc1

v0.3.0 - Beta 2

15 Oct 23:28
f4352b5
Compare
Choose a tag to compare
v0.3.0 - Beta 2 Pre-release
Pre-release

Changes since Beta 1

(also see previous beta 1 for full 0.3.0 changelog)

  • Added better documentation on what loader.dll is and what to do with it; added as a text file asset distributed with the zip
  • Properly set logging level in loader.dll by passing data to the thread. Fixes a weird issue where logging level was not properly set in release mode
  • Fix weird issue in loader.dll with logs suddenly missing if target was set to false.
  • Made logging clearer and better
  • Added log level to config file to allow easily adjusting the level. No need to launch as cli or use an env var anymore to get trace level information
  • Added warn popup to loader.dll if it was not loaded in support application. Reminder that loader.dll is not a native plugin and should not be used as such. It should stay bundled alongside the main exe files.
  • If api call in patch check failed, sleep a little bit to let process catch up before retrying again. If it's still failing after a set timeout period, it's considered a hard error though. (Error in the sense that process injection is aborted; the program will continue working normally)
  • Greatly improved all the warning messages to clearly state what's happening, and that you can press OK and the app continues normally.
  • Lots of general misc fixes and improvements

Full Changelog 0.2.6-b2: v0.2.6...v0.3.0b2
Full Changelog b1-b2: v0.3.0b1...v0.3.0b2

v0.3.0 - Beta 1

15 Oct 21:05
1c3d198
Compare
Choose a tag to compare
v0.3.0 - Beta 1 Pre-release
Pre-release

Notes

This comes with a file loader.dll. No special setup is required. Just keep loader.dll in the same folder as the 2 exe's and everything will work. I have added notes on this in a text file in the zip for the next release to avoid confusion.

Changes

Did a massive refactoring of most of the codebase1. Plugin loading logic was separated out into a separate loader dll to allow for more complicated logic which was not possible (or very very difficult) using the previous method.

  • Faster plugin loading. Once the loader is injected, it does the work of the actual plugin loading at full speed from within the actual process.
  • Also multi-threaded plugin loading to load plugins in parallel. Before this, they only loaded sequentially, so technically it could be slower.
  • The loader now calls any exported Init fn so that any plugin developers can properly program their plugin.2 (P.S. as a plugin dev, your init function can take as long as you want. Though at that point it's probably more recommended to just spawn your own thread so Init can finish in a timely manner)
  • Additionally to the above, developer plugins can also detect when yabg3ml is running vs other mod loaders. This allows you to do things like not run Init code in DllMain, and instead let my mod loader call Init. Otherwise, if it's not mine, you can Init from DllMain to keep compatibility with other mod loaders. See the plugin template for how to do this.
  • Note to plugin devs / mod loader makers: If you were using an old commit of native plugin lib, please update asap. There has been dev work on it and the previous metadata macro is incompatible with this version.
  • native plugin template for plugin devs has been updated and improved. Check it out if you used a previous version.
  • More robust dirty module check, which is accurate cause it checks windows file identifier.
  • Improved process watching code.
  • Use default config values if config is malformed. This will allow it to load if entries are missing, though note if you're doing default values it may load the wrong one since you haven't configured it. Also added popup if loading config failed so it's clearer what happened.
  • Grant SE_DEBUG_NAME if running as admin just to ensure we get extra permissions. This will ensure the program succeeds if it didn't work in non-admin mode
  • General stability improvements due to the refactor and testing.
  • Various ux improvements to not crash but rather display a popup with better information
  • Updated visual style of popups to look more modern.
  • Many many many dependency updates and other misc fixes

Full Changelog: v0.2.6...v0.3.0


  1. There were 49 changed files, 139 commits, and 3,479 additions and 1,242 deletions.

  2. Why? Because actually doing anything non-basic inside DllMain is a very bad idea. Deadlocks, UB (even silent UB), and a whole host of other nasty things can happen if you use DllMain for anything except simple tasks. An Init function is the proper way to do it, so now we support it! This is also reflected in some updates to the plugin template. Also, if you're a dev, the Rust signature is extern "C" fn Init(); if you use C, it is void __cdecl Init(void). See articles below.
    https://devblogs.microsoft.com/oldnewthing/20070904-00/?p=25283
    https://devblogs.microsoft.com/oldnewthing/20040128-00/?p=40853
    https://devblogs.microsoft.com/oldnewthing/20040127-00/?p=40873
    https://devblogs.microsoft.com/oldnewthing/20100115-00/?p=15253
    https://blog.barthe.ph/2009/07/30/no-stdlib-in-dllmai/
    https://learn.microsoft.com/en-us/windows/win32/dlls/dllmain?redirectedfrom=MSDN (see warning section)
    https://learn.microsoft.com/en-us/windows/win32/dlls/dynamic-link-library-best-practices

v0.2.6

21 May 06:02
8bdb1bd
Compare
Choose a tag to compare
  • Fix improper dirty check against <game_dir>/bin/NativeMods folder

Full Changelog: v0.2.5...v0.2.6

v0.2.5

19 May 16:53
93f1e1f
Compare
Choose a tag to compare

What's Changed

  • Tools now automatically detect root bg3 installation path if placed in <bg3_root>/bin or <bg3_root>/bin/subfolder (install_root will not require configuring in this case)
  • Placed zipped files inside bin folder to allow installation via Vortex engine injector
  • Dependency updates

Full Changelog: v0.2.4...v0.2.5

v0.2.4 - More robust error handling

24 Feb 04:47
e36949c
Compare
Choose a tag to compare
  • Even more robust error handling and better user facing messages in case something went wrong
  • More possible sources of crashes eliminated, and in some cases error recovery is possible and handled transparently
  • If winapi still failed to inject into process, user will get a nice popup message instead of a crash (you can just retry again)

Full Changelog: v0.2.3...v0.2.4

v0.2.3 - Crash fixes

21 Feb 00:31
a136fba
Compare
Choose a tag to compare
  • Fix invalid module handle in the process dirty check causing a crash #38
  • Fix case insensitive disabled plugin comparisons a136fba

Full Changelog: v0.2.2...v0.2.3

v0.2.2 - Injection Detection

19 Feb 16:47
b408804
Compare
Choose a tag to compare
  • Fix some more case sensitive comparison parts of file paths
  • Add a bunch of traces + console mode with --cli flag to allow users to print trace output at any time (used for ease of back and forth debugging)
  • Injection detection protects your game from accidents by intelligently aborting plugin injection if game instance has already been patched. This means that you can't accidentally inject plugins into a game instance that's already been injected into. If you added new plugins to your folder while you were playing the game, then you'll need to restart your game before you can reinject all the new plugins.

Full Changelog: v0.2.1...v0.2.2