Co-authored-by: lizzie <lizzie@eden-emu.dev> Reviewed-on: #203 Reviewed-by: Shinmegumi <shinmegumi@eden-emu.dev>
		
			
				
	
	
		
			247 lines
		
	
	
	
		
			9.5 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
			
		
		
	
	
			247 lines
		
	
	
	
		
			9.5 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
| # Development
 | |
| 
 | |
| * **Windows**: [Windows Building Guide](./build/Windows.md)
 | |
| * **Linux**: [Linux Building Guide](./build/Linux.md)
 | |
| * **Android**: [Android Building Guide](./build/Android.md)
 | |
| * **Solaris**: [Solaris Building Guide](./build/Solaris.md)
 | |
| * **FreeBSD**: [FreeBSD Building Guide](./build/FreeBSD.md)
 | |
| * **macOS**: [macOS Building Guide](./build/macOS.md)
 | |
| 
 | |
| # CPM
 | |
| 
 | |
| CPM (CMake Package Manager) is the preferred method of managing dependencies within Eden. Documentation on adding dependencies/using CPMUtil is in the works.
 | |
| 
 | |
| Notes:
 | |
| - `YUZU_USE_CPM` is set by default on MSVC and Android. Other platforms should use this if certain "required" system dependencies (e.g. OpenSSL) are broken or missing
 | |
| - `CPMUTIL_DEFAULT_SYSTEM` can be set to `OFF` to force the usage of bundled dependencies. This can marginally decrease the final package size.
 | |
| - When adding new prebuilt dependencies a la OpenSSL, SDL2, or FFmpeg, there *must* be a CMake option made available to forcefully download this bundle. See the OpenSSL implementation in the root CMakeLists for an example.
 | |
|   * This is necessary to allow for creation of fully-qualified source packs that allow for offline builds after download (some package managers and distros enforce this)
 | |
| 	
 | |
| # Guidelines
 | |
| 
 | |
| ## License Headers
 | |
| All commits must have proper license header accreditation.
 | |
| 
 | |
| You can easily add all necessary license headers by running:
 | |
| ```sh
 | |
| git fetch origin master:master
 | |
| FIX=true COMMIT=true .ci/license-header.sh
 | |
| git push
 | |
| ```
 | |
| 
 | |
| Alternatively, you may omit `COMMIT=true` and do an amend commit:
 | |
| ```sh
 | |
| git fetch origin master:master
 | |
| FIX=true .ci/license-header.sh
 | |
| git commit --amend -a --no-edit
 | |
| ```
 | |
| 
 | |
| ## Pull Requests
 | |
| Pull requests are only to be merged by core developers when properly tested and discussions conclude on Discord or other communication channels. Labels are recommended but not required. However, all PRs MUST be namespaced and optionally typed:
 | |
| ```
 | |
| [cmake] refactor: CPM over submodules
 | |
| [desktop] feat: implement firmware install from ZIP
 | |
| [hle] stub fw20 functions
 | |
| [core] test: raise maximum CPU cores to 6
 | |
| [cmake, core] Unbreak FreeBSD Building Process
 | |
| ```
 | |
| 
 | |
| - The level of namespacing is generally left to the committer's choice.
 | |
| - However, we never recommend going more than two levels *except* in `hle`, in which case you may go as many as four levels depending on the specificity of your changes.
 | |
| - Ocassionally, up to two namespaces may be provided for more clarity.
 | |
|   * Changes that affect the entire project (sans CMake changes) should be namespaced as `meta`.
 | |
| - Maintainers are permitted to change namespaces at will.
 | |
| - Commits within PRs are not required to be namespaced, but it is highly recommended.
 | |
| 
 | |
| # IDE setup
 | |
| 
 | |
| ## VSCode
 | |
| Copy this to `.vscode/settings.json`, get CMake tools and it should be ready to build:
 | |
| ```json
 | |
| {
 | |
|     "editor.tabSize": 4,
 | |
|     "files.watcherExclude": {
 | |
|         "**/target": true
 | |
|     },
 | |
|     "files.associations": {
 | |
|       "*.inc": "cpp"
 | |
|     },
 | |
|     "git.enableCommitSigning": true,
 | |
|     "git.alwaysSignOff": true
 | |
| }
 | |
| ```
 | |
| 
 | |
| You may additionally need the `Qt Extension Pack` extension if building Qt.
 | |
| 
 | |
| # Build speedup
 | |
| 
 | |
| If you have an HDD, use ramdisk (build in RAM):
 | |
| ```sh
 | |
| sudo mkdir /tmp/ramdisk
 | |
| sudo chmod 777 /tmp/ramdisk
 | |
| # about 8GB needed
 | |
| sudo mount -t tmpfs -o size=8G myramdisk /tmp/ramdisk
 | |
| cmake -B /tmp/ramdisk
 | |
| cmake --build /tmp/ramdisk -- -j32
 | |
| sudo umount /tmp/ramdisk
 | |
| ```
 | |
| 
 | |
| ## Debugging (host code)
 | |
| 
 | |
| Ignoring SIGSEGV when debugging in host:
 | |
| 
 | |
| - **gdb**: `handle all nostop pass`.
 | |
| - **lldb**: `pro hand -p true -s false -n false SIGSEGV`.
 | |
| 
 | |
| ## Debugging (guest code)
 | |
| 
 | |
| ### gdb
 | |
| 
 | |
| Run `./build/bin/eden-cli -c <path to your config file (see logs where you run eden normally to see where it is)> -d -g <path to game>`
 | |
| 
 | |
| Then hook up an aarch64-gdb (use `yay aarch64-gdb` or `sudo pkg in arch64-gdb` to install)
 | |
| Then type `target remote localhost:1234` and type `c` (for continue) - and then if it crashes just do a `bt` (backtrace) and `layout asm`.
 | |
| 
 | |
| ### gdb cheatsheet
 | |
| 
 | |
| - `mo <cmd>`: Monitor commands, `get info`, `get fastmem` and `get mappings` are available. Type `mo help` for more info.
 | |
| - `detach`: Detach from remote (i.e restarting the emulator).
 | |
| - `c`: Continue
 | |
| - `p <expr>`: Print variable, `p/x <expr>` for hexadecimal.
 | |
| - `r`: Run
 | |
| - `bt`: Print backtrace
 | |
| - `info threads`: Print all active threads
 | |
| - `thread <number>`: Switch to the given thread (see `info threads`)
 | |
| - `layout asm`: Display in assembly mode (TUI)
 | |
| - `si`: Step assembly instruction
 | |
| - `s` or `step`: Step over LINE OF CODE (not assembly)
 | |
| - `display <expr>`: Display variable each step.
 | |
| - `n`: Next (skips over call frame of a function)
 | |
| - `frame <number>`: Switches to the given frame (from `bt`)
 | |
| - `br <expr>`: Set breakpoint at `<expr>`.
 | |
| - `delete`: Deletes all breakpoints.
 | |
| - `catch throw`: Breakpoint at throw. Can also use `br __cxa_throw`
 | |
| - `br _mesa_error`: Break on mesa errors (set environment variable `MESA_DEBUG=1` beforehand), see [MESA_DEBUG](https://mesa-docs.readthedocs.io/en/latest/debugging.html).
 | |
| 
 | |
| Expressions can be `variable_names` or `1234` (numbers) or `*var` (dereference of a pointer) or `*(1 + var)` (computed expression).
 | |
| 
 | |
| For more information type `info gdb` and read [the man page](https://man7.org/linux/man-pages/man1/gdb.1.html).
 | |
| 
 | |
| # Bisecting older commits
 | |
| 
 | |
| Since going into the past can be tricky (especially due to the dependencies from the project being lost thru time). This should "restore" the URLs for the respective submodules.
 | |
| 
 | |
| ```sh
 | |
| #!/bin/sh -e
 | |
| cat > .gitmodules <<EOF
 | |
| [submodule "enet"]
 | |
| 	path = externals/enet
 | |
| 	url = https://github.com/lsalzman/enet.git
 | |
| [submodule "cubeb"]
 | |
| 	path = externals/cubeb
 | |
| 	url = https://github.com/mozilla/cubeb.git
 | |
| [submodule "dynarmic"]
 | |
| 	path = externals/dynarmic
 | |
| 	url = https://github.com/lioncash/dynarmic.git
 | |
| [submodule "libusb"]
 | |
| 	path = externals/libusb/libusb
 | |
| 	url = https://github.com/libusb/libusb.git
 | |
| [submodule "discord-rpc"]
 | |
| 	path = externals/discord-rpc
 | |
| 	url = https://github.com/yuzu-emu-mirror/discord-rpc.git
 | |
| [submodule "Vulkan-Headers"]
 | |
| 	path = externals/Vulkan-Headers
 | |
| 	url = https://github.com/KhronosGroup/Vulkan-Headers.git
 | |
| [submodule "sirit"]
 | |
| 	path = externals/sirit
 | |
| 	url = https://github.com/yuzu-emu-mirror/sirit.git
 | |
| [submodule "mbedtls"]
 | |
| 	path = externals/mbedtls
 | |
| 	url = https://github.com/yuzu-emu-mirror/mbedtls.git
 | |
| [submodule "xbyak"]
 | |
| 	path = externals/xbyak
 | |
| 	url = https://github.com/herumi/xbyak.git
 | |
| [submodule "opus"]
 | |
| 	path = externals/opus
 | |
| 	url = https://github.com/xiph/opus.git
 | |
| [submodule "SDL"]
 | |
| 	path = externals/SDL
 | |
| 	url = https://github.com/libsdl-org/SDL.git
 | |
| [submodule "cpp-httplib"]
 | |
| 	path = externals/cpp-httplib
 | |
| 	url = https://github.com/yhirose/cpp-httplib.git
 | |
| [submodule "ffmpeg"]
 | |
| 	path = externals/ffmpeg/ffmpeg
 | |
| 	url = https://github.com/FFmpeg/FFmpeg.git
 | |
| [submodule "vcpkg"]
 | |
| 	path = externals/vcpkg
 | |
| 	url = https://github.com/microsoft/vcpkg.git
 | |
| [submodule "cpp-jwt"]
 | |
| 	path = externals/cpp-jwt
 | |
| 	url = https://github.com/arun11299/cpp-jwt.git
 | |
| [submodule "libadrenotools"]
 | |
| 	path = externals/libadrenotools
 | |
| 	url = https://github.com/bylaws/libadrenotools.git
 | |
| [submodule "tzdb_to_nx"]
 | |
| 	path = externals/nx_tzdb/tzdb_to_nx
 | |
| 	url = https://github.com/lat9nq/tzdb_to_nx.git
 | |
| [submodule "VulkanMemoryAllocator"]
 | |
| 	path = externals/VulkanMemoryAllocator
 | |
| 	url = https://github.com/GPUOpen-LibrariesAndSDKs/VulkanMemoryAllocator.git
 | |
| [submodule "breakpad"]
 | |
| 	path = externals/breakpad
 | |
| 	url = https://github.com/yuzu-emu-mirror/breakpad.git
 | |
| [submodule "simpleini"]
 | |
| 	path = externals/simpleini
 | |
| 	url = https://github.com/brofield/simpleini.git
 | |
| [submodule "oaknut"]
 | |
| 	path = externals/oaknut
 | |
| 	url = https://github.com/merryhime/oaknut.git
 | |
| [submodule "Vulkan-Utility-Libraries"]
 | |
| 	path = externals/Vulkan-Utility-Libraries
 | |
| 	url = https://github.com/KhronosGroup/Vulkan-Utility-Libraries.git
 | |
| [submodule "oboe"]
 | |
| 	path = externals/oboe
 | |
| 	url = https://github.com/google/oboe.git
 | |
| [submodule "externals/boost-headers"]
 | |
| 	path = externals/boost-headers
 | |
| 	url = https://github.com/boostorg/headers.git
 | |
| EOF
 | |
| git submodule sync
 | |
| update_or_checkout () {
 | |
|     if [ $0 = 'externals/sirit' ] \
 | |
|     || [ $0 = 'externals/dynarmic' ] \
 | |
|     || [ $0 = 'externals/breakpad' ] \
 | |
|     || [ $0 = 'externals/discord-rpc' ] \
 | |
|     || [ $0 = 'externals/mbedtls' ]; then
 | |
|         [ -f $0/CMakeLists.txt ] || git submodule update --force --remote --init -- $0
 | |
|         echo $0 ':remote' && git submodule update --remote $0
 | |
|         exit
 | |
|     elif [ $0 = 'externals/nx_tzdb/tzdb_to_nx' ]; then
 | |
|         [ -f $0/CMakeLists.txt ] || git submodule update --force --remote --init -- $0
 | |
|         echo $0 ':remote' && git submodule update --remote $0
 | |
|     else
 | |
|         echo $0 ':update' && git submodule update --init $0 && exit
 | |
|         echo $0 ':remote' && git submodule update --remote $0 && exit
 | |
|         echo $0 ':failure'
 | |
|     fi
 | |
| }
 | |
| export -f update_or_checkout
 | |
| grep path .gitmodules | sed 's/.*= //' | xargs -n 1 -I {} bash -c 'update_or_checkout "$@"' {}
 | |
| # Fix for LLVM builds
 | |
| sed -i 's/src\/yuzu\/main.cpp/${CMAKE_SOURCE_DIR}\/src\/yuzu\/main.cpp/g' CMakeModules/FindLLVM.cmake
 | |
| # Only after cloning and everything - fixes issues with Zydis
 | |
| cat > externals/dynarmic/src/dynarmic/common/x64_disassemble.cpp <<EOF
 | |
| #include <cstddef>
 | |
| #include <vector>
 | |
| #include <string>
 | |
| namespace Dynarmic::Common {
 | |
| void DumpDisassembledX64(const void* ptr, size_t size) {}
 | |
| std::vector<std::string> DisassembleX64(const void* ptr, size_t size) { return {}; }
 | |
| }
 | |
| EOF
 | |
| ```
 | |
| 
 | |
| If having issues with older artifacts, then run `rm -r externals/dynarmic/build externals/dynarmic/externals externals/nx_tzdb/tzdb_to_nx/externals externals/sirit/externals`.
 | |
| 
 | |
| Configuring CMake with `-DSIRIT_USE_SYSTEM_SPIRV_HEADERS=1 -DCMAKE_CXX_FLAGS="-Wno-error" -DCMAKE_C_FLAGS="-Wno-error -Wno-array-parameter -Wno-stringop-overflow"` is also recommended.
 |