Tweak-Tutorial

How Do You Create a Tweak?

What is the old ABI?

The ABI that was used by versions of Xcode 11.7 and older, commonly known as “OldABI,” is what was used to compile tweaks prior to iOS 14 releasing. Upon the release of iOS 14 and Xcode 12.0, Apple decided to switch to the new arm64e ABI when a binary is compiled for arm64e.

Tweak developers were advised to compile with the old ABI, however. The reason for this was the fact that the new ABI broke support for arm64e (iPhone Xs and newer) on iOS 13, while the old ABI still worked fine. Apple broke support for the old ABI in iOS 14.5 onward.

The issue with the old ABI on modern iOS

On iOS 14.5-14.8.1, the patches needed to support the old ABI are minimal and are implemented on the jailbeak’s end. Starting with iOS 15, though, arm64e jailbreaks need a more robust solution to support the old ABI, which causes system instability.

The “Legacy arm64e Support” package has been known to cause general system instability, also increasing the chance of spinlock panics, explained here.

How do I use the new ABI in my tweaks?

There are several ways to use the new ABI in your tweaks. We’ll go through them from most to least desirable. Do note that if you would like to support iOS 13 as well as iOS 15+, you will need to compile with the old ABI for “rootful” builds of your tweak and with the new ABI for “rootless” builds.

Compiling via macOS

The best way to make sure the tweak works completely with the new ABI is compiling on a macOS machine. The new ABI has not been open sourced yet as of the time of writing (24 November 2023), therefore it can only be used to compile tweaks on macOS.

Compiling via GitHub Actions

If you do not have access to a macOS machine but would still like to compile with the new ABI, you can use GitHub Actions to do so. GitHub Actions is freely available to any public repository hosted on GitHub and paid for private repositories.

To get started, make a folder in your tweak directory called .github and then a folder called workflows inside of that. Within that folder, create a file ending with the .yml extension. It can be called anything. For instance, the path for the file could be .github/workflows/build.yml. Then, paste the following to the file, graciously provided by @L1ghtmann:

name: Build CI

on:
  push:
  workflow_dispatch:

jobs:
  build:
    runs-on: macos-latest
    steps:
      - uses: actions/checkout@main

      - name: Checkout theos/theos
        uses: actions/checkout@main
        with:
          repository: theos/theos
          ref: master
          submodules: recursive
          path: theos

      - name: Checkout theos/sdks
        uses: actions/checkout@main
        with:
          repository: theos/sdks
          sparse-checkout: iPhoneOS14.5.sdk
          path: theos/sdks

      - name: Build Package
        run: |
          brew install make xz ldid
          export THEOS=theos
          gmake clean package FINALPACKAGE=1
          gmake clean package FINALPACKAGE=1 THEOS_PACKAGE_SCHEME=rootless

      - name: Upload a Build Artifact
        uses: actions/upload-artifact@v3
        with:
          path: packages/*.deb

This is a basic template that should work for most simple tweaks. If the tweak has dependencies, you will need to install them as well.

After pushing the changes, you may or may not have to manually enable GitHub Actions in your repository.

Using Allemand(e)

Another method that can be used to avoid relying on the “Legacy arm64e Support” package is using a static patcher on the binary. There are currently two available: Allemand and Allemande. Allemand supports iOS and macOS, while Allemande is a “port of allemand to C++,” which is cross-platform.

Quick note about static patchers: they are not perfect and may not work 100% of the time. In particular, there are issues with function blocks, so things like animation blocks in UIView will cause the device to crash.

Relying on Legacy arm64e Support

If all else fails, then you can add a dependency on Legacy arm64e Support in your tweak. This is not recommended and should be used as a last resort, because as mentioned previously, the Legacy arm64e Support package causes system instability. The proper way to depend on this is:

Depends: firmware (<< 15.0) | cy+cpu.arm64v8 | oldabi

Note: if you want to support the XinaA15 jailbreak, you will have to remove these dependenices from the “rootful” version of your package and keep them only for rootless.

Previous Page (Adapting for rootless)

Next Page (%hookf)