Merge pull request #14 from suchmememanyskill/tsv3

Tsv3
This commit is contained in:
suchmememanyskill 2021-08-02 00:37:36 +02:00 committed by GitHub
commit 878c334496
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 959 additions and 343 deletions

448
README.md
View file

@ -1,250 +1,254 @@
# TegraScript
The scripting language of TegraExplorer
Notice: TegraScript v2 is entirely different than v1. If you still have v1 scripts, you'll have to rewrite them.
Included in this repository:
- [fwdump.te](https://suchmememanyskill.github.io/TegraScript/scripts/fwDump.te) dumps the firmware with the actual firmware version as it's name
- [systemRestore.te](https://suchmememanyskill.github.io/TegraScript/scripts/systemRestore.te) restores the output of [emmchaccgen](https://github.com/suchmememanyskill/EmmcHaccGen). Place the generated boot.bis and SYSTEM directory next to this script
- [systemwipe.te](https://suchmememanyskill.github.io/TegraScript/scripts/systemwipe.te) is essentially a 'factory reset'. This will wipe everything off the switch and make it boot up like it was fresh out of the box. Error (eprt) reports are kept, so this is not safe to use with sxos.
Any of this scripts can be ran by putting them on the sd, then inside tegraexplorer navigating to the sd, selecting the file and running it.
## General Syntax
### Variables
# Syntax
## Variable assignment
Variables in TegraScript do not need explicit type definitions:
```
variable = function(arg1, arg2) # this calls function with 2 arguments: arg1 and arg2, and stores it in variable
variable = function(arg1, arg2)
```
Variables can be of the following types:
- Integer
- String
- Integer Array
- String Array
- Byte Array
- Empty Array
Creating and accessing Array variables goes as follows:
In TegraScript, user provided functions have no arguments and are written like this:
```
variable = [1,2,3,4,5] # This creates an integer array and stores it into variable
function(variable[2]) # This calls function with 1 argument, index 2 of variable, which is 3
```
In tegrascript, operations are evaluated from left to right. This is important for math type operations. See the operator section for what type definitions you can put operators against. As a quick primer:
```
variable = 2 + 2 * 2 # This puts 8 into variable, as the calculations get evaluated from left to right
variable = 2 + (2 * 2) # But! we can also use brackets to prioritise calculations
variable = "a" + "b" # Adding 2 strings together is also a supported operator
```
Note: Minus integer values are supported this time around
Another note: You can do !variable to flip the integer inside
Another another note: Every object in TScript is not a reference type! every time you try to modify an array it re-makes the array. In practise array operations are slow
Every variable in TegraScript v2 is global, thus you can access any variable in any self-defined function
### Built in variables
TegraScript has 2 built in variables.
- `_CWD`, which is a string, and represents the current working directory, thus the directory the currently running script is in.
- `_EMU`, which is an int, and represents if an emummc is present. 1 for present, 0 for not present
When running `dirRead()` as a function, besides returning a list of filenames, also sets an integer array called `fileProperties` that holds if the representing index of the filenames is a folder or not. 1 for a folder, 0 for a file
### Functions
TegraScript has support for functions
Defining and using a function goes as follows:
```
function = { # We are defining a function called function here, with the {}
# We put the code we want to run inside the function here
var = 1 + 2 + 3
helloWorld = {
print("hello world!")
}
function() # after running this, variable will be set to 6
helloWorld()
```
But you may see an issue: there are no arguments! Fear not, as every variable in TegraScript is global. You can thus solve it with the following syntax:
Arrays in TegraScript can be of 4 types: StringArray, ByteArray, IntegerArray and EmptyArray
```
function = {
b = a * 2 # We want to multiply a by 2 and put it into b, but how do we define a?
strings = ["a", "b", "c"]
numbers = [1,2,3,4,5]
bytes = ["BYTE[]",1,2,3,4,5]
empty = [] # Can be converted to other typed arrays by adding a variable into it
```
## Variable usage
In TegraScript, all variables are global. This means that any variable declared anywhere can be used anywhere
You can use variables in place of static numbers or strings
```
three = 3
1 + 2 + three
```
Most variable types have function members. See below for all types of variables
```
three = 3
three.print()
```
Arrays can be accessed as follows:
(the [arg] syntax is internally converted to variable.get(arg), which is also valid syntax if you wish to use it)
```
numbers = [1,2,3,4,5]
print(numbers[1]) # prints 2
```
When calling a user provided function, everything between the `()` gets evaulated, and the result gets discarded. This means that you can use the `()` to define variables. This used to a TegraScriptv2 bug, but is left in for backwards compatibility
```
a = {
b = b + 1
}
function(a = 10) # Ah! We can just define it in the function args
a = 20
function() # Or if you prefer, you can define it normally as well
a(b = 5)
```
### Flow control
TegraScript has the following functions for flow control: `if()`, `else()`, `while()`, `return()`, `exit()`
- `if()` checks it's first arg, if it's 1, it runs the next {}, otherwise it skips over it
- `else()` checks if the last statement was an if, and if that if skipped, we run the next {}
- `while()` checks it's first arg, if it's 1, it runs the next {}, and jumps back to the while
- `return()` exits the current running function, if there is one
- `exit()` exits the script entirely
Let's try to build a loop that iterates 30 times, and printing even numbers
Although normal assignment will work too
```
i = 0
while (i < 30){ # check if i is below 30, if so, run the code between the brackets
if (i % 2 == 0){ # is i dividable by 2?
println(i)
}
i = i + 1 # don't forget the + 1 of i! otherwise an infinite loop will haunt you
a = {
b = b + 1
}
b = 5
a()
```
## Operators
### Integer, Integer
Operator | Output
|:-|:-|
`+` | sum of both integers
`-` | integer minus integer
`*` | multiplication of both integers
`/` | integer divided by integer
`%` | integer division remainder (modulo) of integer
`<` | 1 if left is smaller, otherwise 0
`>` | 1 if left is bigger, otherwise 0
`<=`| 1 if left is smaller or equal, otherwise 0
`>=`| 1 if left is bigger or equal, otherwise 0
`==`| 1 if left and right are equal, otherwise 0
`!=`| 1 if left and right are not equal, otherwise 0
`&&`| 1 if left and right are non 0, otherwise 0. Also if output is 0, disregards rest of statement
`\|\|`| 1 if left or right are non 0, otherwise 0. Also if output is 1, disregards rest of statement
`&` | Binary operator. ANDs both integers together
`\|` | Binary operator. ORs both integers together
`<<`| Binary operator. Bitshifts the left integer to the left by right integer's amount
`>>`| Binary operator. Bitshifts the left integer to the right by right integer's amount
### String, String
Operator | Output
|:-|:-|
`+` | Adds both strings together
`==`| 1 if strings are equal, otherwise 0
`-` | Removes the end of the first string if the second string matches the end of the first. (Example: `"abcdef" - "def"` is `"abc"`)
`/` | Splits the left string based on the right string. Returns a string array
### (Integer Array, Byte Array), Integer
Operator | Output
|:-|:-|
`+` | Adds the right integer into the left array
`-` | Removes right integer amount of entries from the left array
`:` | Removes right integer amount of entries from the beginning of the left array
### String, Integer
Operator | Output
|:-|:-|
`-` | Removes right integer amount of characters from the left string
`:` | Removes right integer amount of character from the beginning of the left string
### StringArray, String
Operator | Output
|:-|:-|
`+` | Adds a string to an array
### EmptyArray, (String, Int)
Operator | Output
|:-|:-|
`+` | Creates the array of the right type and puts the given value as the first entry
## Functions
### Flow control functions
Name | Description | OutType
|:-|:-|:-|
`if(int arg)` | Runs the next {} if arg is non zero | None
`else()` | Runs the next {} if the last statement was an if that skipped their {} | None
`while(int arg)` | Runs the next {} if arg is non zero. Jumps to the if after exiting the {} | None
`return()` | Breaks out of a function | None
`exit()` | Exits the current script | None
### Utilities
Name | Description | OutType
|:-|:-|:-|
`print(...)` | Prints all args provided to the screen. Can print Integer, String, IntegerArray. `\r` and `\n` is supported | None
`println(...)` | Same as `print(...)` but puts a newline at the end | None
`color(string color)`| Sets the print color. Supported inputs: `"RED"`, `"ORANGE"`, `"YELLOW"`, `"GREEN"`, `"BLUE"`, `"VIOLET"`, `"WHITE"` | None
`len(var arg1)` | Gets the length of a string or an array | Integer
`byte(IntegerArray arg1)`| Converts an integer array to a byte one | ByteArray
`bytesToStr(ByteArray arg1)`| Converts a byte array to a string | String
`printPos(int x, int y)` | Sets the printing position on screen. X/Y are in whole character sizes (16x16) | None
`clearscreen()` | Clears the screen full of your nonsense | None
`drawBox(int x1, int y1, int x2, int y2, int color)` | Draws a box from x1/y1 to x2/y2 with the color as color (raw: 0x00RRGGBB) | None
`wait(int ms)` | Waits for ms amount | None
`pause()` | Pauses until controller input is detected. Returns the controller input as raw u32 bitfield | Integer
`version()` | Returns an Integer array of the current TE version | IntegerArray
`menu(StringArray options, int startPos)` | Makes a menu with the cursor at startPos, with the provided options. Returns the current pos when a is pressed. B always returns 0 | Integer
`menu(StringArray options, int startPos, StringArray colors)` | Same as above, but the entries now get colors defined by the colors array. Uses the same colors as the `colors()` function | Integer
`menu(StringArray options, int startPos, StringArray colors, IntegerArray modifier)` | Same as above, but entries can be hidden or skipped defined by the modifier array. 0 for normal, 1 for skip, 2 for hide | Integer
Note about `pause()`. You need to work with raw bitfields. Have an example
You can also invert integers as follows:
```
# The most common controls
JoyY = 0x1
JoyX = 0x2
JoyB = 0x4
JoyA = 0x8
!1 # becomes 0
!0 # becomes 1
LeftJoyDown = 0x10000
LeftJoyUp = 0x20000
LeftJoyRight = 0x40000
LeftJoyLeft = 0x80000
if (pause() & JoyX){
println("X has been pressed!")
}
1.not() # becomes 0
```
### FileSystem functions
Name | Description | OutType
|:-|:-|:-|
`fileRead(string path)` | Reads the file at the given path and returns it's contents in a byte array | ByteArray
`fileWrite(string path, ByteArray data)` | Writes data to the given path. Returns non zero on error | Integer
`fileExists(string path)` | Checks if a file or folder exists at the given path. 1 if yes, otherwise 0 | Integer
`fileMove(string src, string dst)` | Moves a file from src to dst. Returns non zero on error | Integer
`fileCopy(string src, string dst)` | Copies a file from src to dst. Returns non zero on error | Integer
`fileDel(string path)` | Deletes the file located at path. Returns non zero on error | Integer
`pathCombine(...)` | Needs 2+ string args as input. Combines them into a path. First entry must be the source folder Example: `pathCombine("sd:/", "tegraexplorer")` -> `"sd:/tegraexplorer"` | String
`pathEscFolder(string path)`| Escapes a folder path. Example: `pathEscFolder("sd:/tegraexplorer")` -> `"sd:/"` | String
`dirRead(string path)` | Reads a folder and returns a StringArray of filenames. Also creates an IntegerArray called `fileProperties` that is the same length as the filenames, and is non zero if the index of the filename is a folder | StringArray
`dirCopy(string src, string dst)`| Copies a folder from src to dst. Dst needs to be the containing folder of where you want src to go (`"sd:/tegraexplorer", "sd:/backup` -> `"sd:/backup/tegraexplorer"`). Returns non zero on error | Integer
`dirDel(string path)` | Deletes the dir located at path. Returns non zero on error | Integer
`mkdir(string path)` | Makes a directory at path | Integer
`launchPayload(string path)`| Launches the payload at the given path. Returns non zero on error | Integer
## Operator precedence
TegraScript has no operator precedence. It runs operators from left to right, always. You can put code inbetween `()` to calculate it before it is used.
```
println(2 + 2 * 2) # prints 8
println(2 + (2 * 2)) # prints 6
```
### Storage functions !! Dangerous
Name | Description | OutType
|:-|:-|:-|
`mmcConnect(string loc)` | Loc can be `"SYSMMC"` or `"EMUMMC"`. Returns non zero on error | Integer
`mmcMount(string loc)` | Loc can be `"PRODINFOF"`, `"SAFE"`, `"SYSTEM"` and `"USER"`. Mounts the filesystem to the prefix `bis:/`. Returns non zero on error | Integer
`mmcDump(string path, string target)`| Dumps target to path. target can be anything in the EMMC menu in TE. Returns non zero on error | Integer
`mmcRestore(string path, string target, int force)`| Restores path to target. target can be anything in the EMMC menu in TE. Force forces smaller files to flash anyway. Returns non zero on error | Integer
`ncaGetType(string path)`| Returns the type of nca that is in path. Make sure you provide an nca and nothing else! Useful for differentiating between Meta and Non-Meta Nca's | Integer
`saveSign(string path)` | Signs the (system) save at the given location. Make sure you provide a save and nothing else! Returns non zero on error | Integer
## Flow control
TegraScript has the following functions for flow control:
- `if(int)` Checks if the integer is non-0, then executes the user provided function behind it. Example: `if (1) { print("Hello world!") }`.
- if returns an elseable class. You can call `.else()` on it to get an else. Example: `if (0) { } .else() { print("hi") }`
- `while(int)` Checks if the integer is non-0, then keeps executing the user provided function behind it until the integer is 0. Example: `i = 0 while (i < 10) { print(i) i = i + 1 }`
- `break()` Raises an exception that is caught by while and foreach loops. Example: `while (1) { break() } # This will end`
- `exit()` Raises an exception to exit the script
# Changelog
### 11/1/2021 @ 22:10 // Wow it isn't 2am for once!
No this is not a sensible language, thanks for asking
For the release of TE 3.0.1, the following changes were made:
- A new type was added: Empty array type. You can get this type by running `var = []`
- Operators were added for StringArray, String and EmptyArray, (String, Integer)
- launchPayload was added as a function
- Menu now allows for an additional integer array to define hidden or skip flags
- A bug was fixed with <= and >= registering as < and >
- fileCopy's should not leave an ugly [ anymore
- A script now get executed at TE bootup if a script called `startup.te` is found on the root of the SD
# Comments
Comments in TegraScript are started with `#`, and last until the end of the line.
### 5/1/2021 @ 1:32am // Sleep is still a lie
There are some special comments to aid in prerequisites.
- `#REQUIRE VER x.x.x` Requires a minimum TegraScript version to run the script. x.x.x should be the minimum version, so like `#REQUIRE VER 4.0.0`
- `#REQUIRE MINERVA` Requires extended memory to run the script. This should be used if you work with large files, saves or if your script is generally very big (20kb+)
- `#REQUIRE KEYS` Requires keys to be dumped to run the script
- `#REQUIRE SD` Requires the sd to be mounted to run the script
- Initial writeup on TScript v2
# Classes
## Integer Class
Integers in TegraScript are always 8 bytes in length
Static defenition example:
```
10 # 10
0x10 # 16
```
| Operator | RVal | Result | Description |
|----------|---------|---------|-----------------------|
| + | Integer | Integer | Addition |
| - | Integer | Integer | Subtraction |
| * | Integer | Integer | Multiplication |
| / | Integer | Integer | Division |
| % | Integer | Integer | Modulo |
| < | Integer | Integer | Smaller than |
| > | Integer | Integer | Bigger than |
| <= | Integer | Integer | Smaller or equal than |
| >= | Integer | Integer | Bigger or equal than |
| == | Integer | Integer | Equal to |
| != | Integer | Integer | Not equal to |
| && | Integer | Integer | Logical AND |
| \|\| | Integer | Integer | Logical OR |
| & | Integer | Integer | And |
| \| | Integer | Integer | Or |
| << | Integer | Integer | Bitshift left |
| Member | Arguments | Argument descriptions | Result | Description |
|---------|-----------|-----------------------|---------|----------------------------------|
| print() | - | - | None | Prints the integer |
| not() | - | - | Integer | Inverts the integer |
| str() | - | - | String | Converts the integer to a string |
## String Class
Strings in TegraScript are immutable
Static defenition example:
```
"Hello" # Hello as text
"\r\n" # Return to left side of the screen, newline
```
| Operator | RVal | Result | Description |
|----------|---------|--------------|---------------------------------------|
| + | String | String | Adds 2 strings together |
| - | Integer | String | Reduces string size by integer amount |
| == | String | Integer | Checks string equality |
| != | String | Integer | Checks string inequality |
| / | String | String Array | Splits left string every right string |
| Member | Arguments | Argument descriptions | Result | Description |
|---------|-----------|-----------------------|--------------|--------------------------------------|
| print() | - | - | None | Prints the string |
| len() | - | - | Integer | Gets the length of the string |
| bytes() | - | - | Byte Array | Returns the string as a byte array |
| get() | Integer | 0: Index of character | String | Returns the character at given index |
| split() | String | 0: String to split on | String Array | Splits a string |
## Array class
Array classes can be of 4 types: Integer array, string array, byte array and empty array
An empty array will be converted into an integer or string array if an integer or string gets added.
You can add integers to byte arrays, and internally they'll be converted to bytes before being added
ArrayReference: arrayReference classes contain a .project() function. This returns the array. You can project anytime, but do note the result is not memory safe and not garuanteed to be valid after changing the array. It's best to re-project before every use.
You can get an index from the array by using `array[index]`, Example: `[1,2,3][0]`
You can also set an array index by using `array[index] = variable`. Example: `a = [1,2,3].copy() a[0] = 69`
Arrays that are static `[1,2,3]` are read-only and will need to be .copy()'d before modification, because they are evaluated before runtime. Arrays that are not static (`[a,b,c]`) will be evaluated every time the script runs past it, and thus do not need to be copied to be modified
| Operator | RVal | Result | Description |
|----------|-------------------------|---------|-----------------------------------|
| + | (Dependant on type) | None | Adds type to array |
| - | Integer | None | Removes integer length from array |
| == | ByteArray, IntegerArray | Integer | Compares 2 number arrays |
| Member | Arguments | Argument descriptions | Result | Description |
|--------------|----------------------------|------------------------------|----------------------------|------------------------------------------------------------------------------------------------------------------------|
| get() | Integer | 0: index of array | String or Integer | Gets a variable at index. You can also use array[index] |
| len() | - | - | Integer | Get the length of an array |
| slice() | Integer, Integer | 0: skip, 1: take | ArrayReference (see above) | Gets a read-only slice of the array |
| foreach() | String | 0: Name of variable | None | Foreaches trough the array using the `{}` behind the foreach. Sets the variable in the string to the iterated variable |
| copy() | - | - | Array | Copy's the array. Required to edit read-only arrays |
| set() | Integer, String or Integer | 0: index, 1: variable to set | None | Sets the variable at array index. You can also use array[index] = var |
| add() | (Dependant on type) | 0: to add | None | Adds type to array |
| contains() | (Dependant on type) | 0: to check if contains | Integer | Checks if array contains item. 0 if false, 1 if true |
| bytestostr() | - | - | String | Converts a byte array to string |
| find() | (Same as type) | 0: needle | Integer | Searches needle in haystack. Returns offset, or -1 if not found |
## Dictionary class
You can initiate a dictionary by using `dict()`. This will create an empty dictionary. You can assign to a dictionary by doing the following: `a = dict() a.b = 1` After this, you can access the variables you put in by doing the following: `print(a.b)` This class is used as returns in some functions in the standard library
## Save class
You can initiate a save class by using `readsave("path/to/save")`. This uses a lot of memory.
| Member | Arguments | Argument descriptions | Result | Description |
|----------|-------------------|--------------------------------|-----------|---------------------------------------------------------------------------|
| read() | String | 0: Path | ByteArray | Reads a file inside the save |
| write() | String, ByteArray | 0: Path, 1: ByteArray to write | Integer | Writes to a file inside the save. Returns 0 on success, non zero on error |
| commit() | - | - | None | Flushes the writes made to save and signs the save |
# Standard library
| Function | Arguments | Argument descriptions | Result | Description | Example |
|---|---|---|---|---|---|
| if | Integer | 0: Execute if not 0 | Elseable (.else(){}) | Executes function behind if, if the integer is non 0 | if (1) { print("Hello world!") } |
| while | Integer | 0: Execute if not 0 | None | Executes function behind while as long as integer is non 0 | i = 0 while (i < 10) { println(i) i = i + 1 } |
| exit | - | - | None | Exits the script | exit() |
| break | - | - | None | Exits a loop, or the script if not in a loop | while (1) { break() } println("The loop broke") |
| readsave | String | 0: Path to save | Save | Opens a save. Causes a fatal error if the save failed to open | mountsys("SYSTEM") readsave("bis:/save/8000000000000120").commit() |
| dict | - | - | Dictionary | Creates an empty dictionary | a = dict() |
| print | Any arg count of (Integer, String) | any: To be printed | None | Calls .print() on every argument. Puts spaces inbetween each argument | print("Hello world!") |
| println | Any arg count of (Integer, String) | any: To be printed | None | Calls print, then prints a newline | println("Hello world") |
| printpos | Integer, Integer | 0: X position, range 0-78, 1: Y position, range 0-42 | None | Sets the print position | printpos(20,20) |
| setpixel | Integer, Integer, Integer | 0: X position, range 0-1279, 1: Y position, range 0-719, 2: Hex color (0xRRGGBB) | None | Sets a pixel on the screen to the desired color | setpixel(0,0,0xFFFFFF) |
| setpixels | Integer, Integer, Integer, Integer, Integer | 0: X0 position, 1: Y0 position, 2: X1 position, 3: Y1 position, 4: Hex color | None | Sets a bunch of pixels on the screen to the desired color in the user provided ranges | setpixels(0,0,1279,719,0xFF0000) |
| emu | - | - | Integer | Returns if an emummc is enabled. returns 1 if enabled | print(emu()) |
| cwd | - | - | String | Returns the current working directory. Returns 'sd:/' if ran from builtin script | print(cwd()) |
| clear | - | - | None | Clears the screen and resets the cursor to the top left | clear() |
| timer | - | - | Integer | Get the current system time in ms | a = timer() println("Very intensive operation") print("Time taken:", timer() - a) |
| pause | - | - | Dictionary (a, b, x, y, down, up, right, left, power, volplus, volminus, raw) | Waits for user input, then returns a dictionary of what was pressed | p = pause() if (p.a) { println("A was pressed!") } |
| pause | Integer | 0: Bitmask | Dictionary (a, b, x, y, down, up, right, left, power, volplus, volminus, raw) | Same as argless pause() but you can provide your own bitmask. See source/hid/hid.h | pause(0xC) # Waits until either A or B is pressed |
| color | Integer | 0: Hex color (0xRRGGBB) | None | Sets the text color | color(0x00FF00) print("Green text!") |
| menu | StringArray, Interger | 0: Menu options, 1: Starting position | Integer | Makes a menu of the strings in the array. Returns the index of what was pressed. Pressing b always returns 0 | idx = menu(["1", "2", "3"], 0) println(idx, "was pressed!") |
| menu | StringArray, Integer, IntegerArray | 0: Menu options, 1: Starting position, 2: Color/Options array | Integer | Makes a menu. The lower 3 bytes of the 3rd argument is RGB (0xRRGGBB). The upper byte is a bitfield with: 0x1: skip, 0x2: hide, 0x4: fileIcon, 0x8: folderIcon | menu(["1", "2"], 0, [0xFF0000, 0x00FF00]) |
| power | Integer | 0: https://github.com/suchmememanyskill/TegraExplorer/blob/master/bdk/utils/util.h#L26 | None | Sets the power state. This is an advanced function. | power(3) # Powers off and resets regulators |
| mountsys | String | 0: Mount target | Integer | Mounts a partition on sys/internal storage. Returns 0 on success, 1 on failure | mountsys("SYSTEM") |
| mountemu | String | 0: Mount target | Integer | Same as mountsys, but on emummc instead. | mountemu("SYSTEM") |
| ncatype | String | 0: Path to nca | Integer | Returns the nca type. Returns -1 on error. See | - |
| emmcread | String, String | 0: File destination, 1: Partition on sys | Integer | Dumps the provided partition to a file. Returns 0 on success, non zero on failure | emmcread("sd:/PRODINFO", "PRODINFO") |
| emmcwrite | String, String | 0: File source, 1: Partition on sys | Integer | Restores the file to the provided partition. Returns 0 on success, non zero on failure | emmcwrite("sd:/PRODINFO", "PRODINFO") |
| emummcread | String, String | 0: File destination, 1: Partition on emu | Integer | Same as emmcread() | emummcread("sd:/PRODINFO", "PRODINFO") |
| emummcwrite | String, String | 0: File source, 1: Partition on emu | Integer | Same as emmcwrite() | emummcwrite("sd:/PRODINFO", "PRODINFO") |
| readdir | String | 0: Path to dir | Dictionary (result, files, folders, fileSizes) | Reads a directory. Puts all files in files (StringArray), folders in folders (StringArray), and file sizes in fileSizes (IntegerArray). .result is an Integer, and 0 on success | a = readdir("sd:/") if (a.result) { exit() } a.files.foreach("x") { println(x) } |
| deldir | String | 0: Path to dir | Integer | Deletes a directory. Prints whatever it's deleting. Returns 0 on success | deldir("sd:/sxos") |
| mkdir | String | 0: Path to new dir | Integer | Creates a new directory. Returns 0 on success | mkdir("sd:/tegraexplorer") |
| copydir | String, String | 0: Path to dir, 1: Path to dir where the dir will be copied into | Integer | Copies a folder into another folder. Returns 0 on success | copydir("sd:/tegraexplorer", "sd:/switch") # This will copy sd:/tegraexplorer to sd:/switch/tegraexplorer |
| copyfile | String, String | 0: Path to source file, 1: Path to dest file | Integer | Copies source file to destination. Returns 0 on success | copyfile("sd:/hbmenu.nro", "sd:/hbmenu_2.nro") |
| movefile | String, String | 0: Source Path, 1: Destination path | Integer | Renames or moves the file (or folder) to destination. Returns 0 on success | movefile("sd:/hbmenu.nro", "sd:/lol.nro") |
| delfile | String | 0: Path to file | Integer | Deletes a file. Returns 0 on success | delfile("sd:/boot.dat") |
| readfile | String | 0: Path to file | ByteArray | Reads a file and returns a byte array. This could be a memory intensive operation. Raises a fatal error if there was an error reading the file | a = readfile("sd:/hbmenu.nro") |
| writefile | String, ByteArray | 0: Path to file, 1: Bytes to write | Integer | Writes bytes to a file. Returns 0 on success | writefile("sd:/yeet", ["BYTE[]", 1,2,3]) |
| fsexists | String | 0: Path | Integer | Checks if a file or folder exists. Returns 1 when it exists, 0 if it does not | fsexists("sd:/hbmenu.nro") |
| payload | String | 0: Path to payload | Integer | Boots to a payload. Returns an integer on failure | payload("sd:/atmosphere/reboot_payload.bin") |
| combinepath | Any arg count of (String) | any: String to be combined | String | Combines path parts together into a path | combinepath("sd:/", "tegraexplorer", "firmware") # Returns "sd:/tegraexplorer/firmware" |
| escapepath | String | 0: String to escape a folder | String | Escapes a folder path (to go 1 folder back). See example | escapepath("sd:/tegraexplorer/firmware") # Returns "sd:/tegraexplorer" |

53
scripts/Dump_saves.te Normal file
View file

@ -0,0 +1,53 @@
#REQUIRE SD
#REQUIRE KEYS
#REQUIRE VER 4.0.0
partNames = ["SYSTEM", "USER"]
menuOptions = ["Exit"].copy()
partNames.foreach("x"){
menuOptions.add("Sysmmc " + x)
}
if (emu()){
menuOptions.add("Emummc " + x)
}
res = menu(menuOptions, 0)
if (!res){
exit()
}
clear()
if (res % 2) {
mountStr = "SYSTEM"
}.else() {
mountStr = "USER"
}
if (res > 2) {
mount = mountemu
}.else() {
mount = mountsys
}
if (mount(mountStr)){
println("Mounting failed!")
pause()
exit()
}
cpSaveFolder = menuOptions[res]
cpSavePath = combinepath("sd:/tegraexplorer/", cpSaveFolder)
mkdir("sd:/tegraexplorer")
mkdir(cpSavePath)
res = copydir("bis:/save", cpSavePath)
if (res) {
println("Copy failed! Errcode: ", res)
}.else() {
println("Done!")
}
pause()

309
scripts/SystemRestoreV3.te Normal file
View file

@ -0,0 +1,309 @@
#REQUIRE VER 4.0.0
#REQUIRE MINERVA
#REQUIRE KEYS
#REQUIRE SD
wait={t=timer()while(timer()<(t+tw)){print("Wait for",(t+tw-timer()/1000),"seconds \r")}}
p = println
getInt = {
resInt = (buff[0] << 24) | (buff[1] << 16) | (buff[2] << 8) | (buff[3])
}
bisFileOffset = 0
getBisBytes = {
buff = bisFileDump.slice(bisFileOffset, bisGetLen).project()
bisFileOffset = bisFileOffset + bisGetLen
}
getNextInt = {
getBisBytes(bisGetLen = 4)
getInt()
}
fatal = {
color(0xFF0000)
p("\n[FATAL]", fatalMsg)
pause()
exit()
}
readBisAttrs = {
path = combinepath(cwd(), "boot.bis")
bisFileRes = 1
if (fsexists(path)){
bisFileDump = readfile(path)
getBisBytes(bisGetLen = 0x10)
bisName = buff.bytestostr()
getBisBytes(bisGetLen = 1)
sizes = [0,1,2,3].copy()
sizes.foreach("x"){
getNextInt()
sizes[x] = resInt
}
bisFileRes = 0
}
}
extractBis = {
color(0xFF0000)
p("Extracting boot.bis...")
color(0xFFFFFF)
i = 0
files = ["BOOT0.bin", "BOOT1.bin", "BCPKG2-1-Normal-Main", "BCPKG2-3-SafeMode-Main"]
sizes.foreach("x"){
getBisBytes(bisGetLen = x)
p("Extracting", files[i])
if (writefile(combinepath(cwd(), files[i]), buff)){
fatal(fatalMsg = "Extracting boot.bis failed!")
}
i = i + 1
}
res = copyfile(combinepath(cwd(), "BCPKG2-1-Normal-Main"), combinepath(cwd(), "BCPKG2-2-Normal-Sub"))
if (!res){
res = copyfile(combinepath(cwd(), "BCPKG2-3-SafeMode-Main"), combinepath(cwd(), "BCPKG2-4-SafeMode-Sub"))
}
if (res) {
fatal(fatalMsg = "Extracting boot.bis failed!")
}
p("\n")
}
bisFiles = ["BOOT0.bin", "BOOT1.bin", "BCPKG2-1-Normal-Main", "BCPKG2-2-Normal-Sub", "BCPKG2-3-SafeMode-Main", "BCPKG2-4-SafeMode-Sub", "PRODINFO"]
backupBis = {
color(0xFF0000)
p("Backing up BIS...")
color(0xFFFFFF)
mkdir(combinepath(cwd(), "backup"))
path = combinepath(cwd(), "backup", "bis")
mkdir(path)
bisFiles.foreach("x"){
print("Backing up", x, "")
if (mmcread(combinepath(path, x), x)){
fatal(fatalMsg = "Failed backing up BIS")
}
p()
}
p()
}
backupSys = {
color(0xFF0000)
p("Backing up SYSTEM...")
color(0xFFFFFF)
mkdir(combinepath(cwd(), "backup"))
path = combinepath(cwd(), "backup", "system")
mkdir(path)
if (mount("SYSTEM")){
fatal(fatalMsg = "Failed to mount SYSTEM")
}
if (copydir("bis:/Contents/registered", path)){
fatal(fatalMsg = "Failed to copy registered")
}
if (copyfile("bis:/save/8000000000000120", combinepath(path, "8000000000000120"))){
fatal(fatalMsg = "Failed to copy 120 save")
}
p("\n")
}
signSave = {
color(0xFF0000)
p("Signing save...")
color(0xFFFFFF)
if (mount("SYSTEM")){
fatal(fatalMsg = "Failed to mount SYSTEM")
}
sav = readsave("bis:/save/8000000000000120")
if (sav.commit()){
fatal(fatalMsg = "Failed to sign save!")
}
sav = 0 # Free sav
}
writeBis = {
color(0xFF0000)
p("Writing BIS...")
color(0xFFFFFF)
toRestore = bisFiles.copy()
toRestore - 1
toRestore.foreach("x"){
print("Restoring", x + "... ")
if (mmcwrite(combinepath(cwd(), x), x)){
fatal(fatalMsg = "BIS write failed!!!")
}
p()
}
p()
}
writeSys = {
if (mount("SYSTEM")){
fatal(fatalMsg = "Failed to mount SYSTEM")
}
color(0xFF0000)
print("Deleting SYSTEM contents... ")
color(0xFFFFFF)
if (deldir("bis:/Contents/registered")){
fatal(fatalMsg = "An error occured during SYSTEM deletion!")
}
p()
color(0xFF0000)
print("Copying registered... ")
color(0xFFFFFF)
if (copydir(combinepath(cwd(), "SYSTEM/Contents/registered"), "bis:/Contents")) {
fatal(fatalMsg = "An error occured during SYSTEM copying!")
}
p()
color(0xFF0000)
print("Copying 120 save... ")
color(0xFFFFFF)
if (copyfile(combinepath(cwd(), "SYSTEM/save/8000000000000120"), "bis:/save/8000000000000120")) {
fatal(fatalMsg = "Failed to copy system save")
}
p("\n")
}
actionRestoreBis = {
extractBis()
backupBis()
writeBis()
}
actionRestoreSys = {
backupSys()
writeSys()
signSave()
}
readBisAttrs()
bootBisFound = !bisFileRes
systemFolderFound = fsexists(combinepath(cwd(), "SYSTEM"))
if ((!bootBisFound) && (!systemFolderFound)){
fatal(fatalMsg = "Nothing to restore found...\nPut a boot.bis and a SYSTEM dir from emmchaccgen next to this script!")
}
if (!bootBisFound){
p("Boot.bis not found. Cannot restore bis. Press any key to continue...")
pause()
}
if (!systemFolderFound){
p("SYSTEM dir not found. Cannot restore SYSTEM. Press any key to continue...")
pause()
}
useSys = 1
if (emu()){
p("Switch System Restore\n")
res = menu(["Exit", "Apply on Sysmmc", "Apply on Emummc"], 0)
if (res == 0){
exit()
}
if (res == 2){
useSys = 0
}
}
if (useSys){
mmcread = emmcread
mmcwrite = emmcwrite
mount = mountsys
}.else() {
mmcread = emummcread
mmcwrite = emummcwrite
mount = mountemu
}
clear()
p("Switch System Restore")
if (bootBisFound){
p("Target:", bisName)
}
p()
res = menu(["Exit", "Restore BIS", "Restore SYSTEM", "Restore Both"], 0)
if (res == 0){
exit()
}
clear()
restoreBis = 0
restoreSystem = 0
if ((res == 1) || (res == 3)){
if (!bootBisFound){
fatal(fatalMsg = "Boot.bis not found...")
}
restoreBis = 1
}
if ((res == 2) || (res == 3)){
if (!systemFolderFound){
fatal(fatalMsg = "SYSTEM dir not found...")
}
restoreSystem = 1
}
clear()
p("Switch System Restore")
color(0xFF00)
if (useSys){
mmcStr = "Sysmmc"
}.else() {
mmcStr = "Emummc"
}
p("Restore BIS:", restoreBis)
p("Restore SYSTEM:", restoreSystem)
p("Restoring onto:", mmcStr)
color(0xFFFFFF)
p("Ready!")
color(0xE70000)
print("WARNING!!!\nIf you do not know exactly what this does. STOP!!!\nThis will fuck with your system!\nOnly do this as a last ditch recovery effort!\n")
color(0x40FF00)
p("For help go here: https://discord.gg/C29hYvh\n")
color(0xFFFFFF)
wait(tw = 10000)
p("Press Power to start the restore, any other key to exit")
if (!pause().power){
exit()
}
clear()
startMs = timer()
if (restoreBis){
actionRestoreBis()
}
if (restoreSystem){
actionRestoreSys()
}
p("\n\nFully done in", (timer() - startMs) / 1000 ,"seconds!\nPress any key to exit")
pause()

250
v2/README.md Normal file
View file

@ -0,0 +1,250 @@
# TegraScript
The scripting language of TegraExplorer
Notice: TegraScript v2 is entirely different than v1. If you still have v1 scripts, you'll have to rewrite them.
Included in this repository:
- [fwdump.te](https://suchmememanyskill.github.io/TegraScript/scripts/fwDump.te) dumps the firmware with the actual firmware version as it's name
- [systemRestore.te](https://suchmememanyskill.github.io/TegraScript/scripts/systemRestore.te) restores the output of [emmchaccgen](https://github.com/suchmememanyskill/EmmcHaccGen). Place the generated boot.bis and SYSTEM directory next to this script
- [systemwipe.te](https://suchmememanyskill.github.io/TegraScript/scripts/systemwipe.te) is essentially a 'factory reset'. This will wipe everything off the switch and make it boot up like it was fresh out of the box. Error (eprt) reports are kept, so this is not safe to use with sxos.
Any of this scripts can be ran by putting them on the sd, then inside tegraexplorer navigating to the sd, selecting the file and running it.
## General Syntax
### Variables
Variables in TegraScript do not need explicit type definitions:
```
variable = function(arg1, arg2) # this calls function with 2 arguments: arg1 and arg2, and stores it in variable
```
Variables can be of the following types:
- Integer
- String
- Integer Array
- String Array
- Byte Array
- Empty Array
Creating and accessing Array variables goes as follows:
```
variable = [1,2,3,4,5] # This creates an integer array and stores it into variable
function(variable[2]) # This calls function with 1 argument, index 2 of variable, which is 3
```
In tegrascript, operations are evaluated from left to right. This is important for math type operations. See the operator section for what type definitions you can put operators against. As a quick primer:
```
variable = 2 + 2 * 2 # This puts 8 into variable, as the calculations get evaluated from left to right
variable = 2 + (2 * 2) # But! we can also use brackets to prioritise calculations
variable = "a" + "b" # Adding 2 strings together is also a supported operator
```
Note: Minus integer values are supported this time around
Another note: You can do !variable to flip the integer inside
Another another note: Every object in TScript is not a reference type! every time you try to modify an array it re-makes the array. In practise array operations are slow
Every variable in TegraScript v2 is global, thus you can access any variable in any self-defined function
### Built in variables
TegraScript has 2 built in variables.
- `_CWD`, which is a string, and represents the current working directory, thus the directory the currently running script is in.
- `_EMU`, which is an int, and represents if an emummc is present. 1 for present, 0 for not present
When running `dirRead()` as a function, besides returning a list of filenames, also sets an integer array called `fileProperties` that holds if the representing index of the filenames is a folder or not. 1 for a folder, 0 for a file
### Functions
TegraScript has support for functions
Defining and using a function goes as follows:
```
function = { # We are defining a function called function here, with the {}
# We put the code we want to run inside the function here
var = 1 + 2 + 3
}
function() # after running this, variable will be set to 6
```
But you may see an issue: there are no arguments! Fear not, as every variable in TegraScript is global. You can thus solve it with the following syntax:
```
function = {
b = a * 2 # We want to multiply a by 2 and put it into b, but how do we define a?
}
function(a = 10) # Ah! We can just define it in the function args
a = 20
function() # Or if you prefer, you can define it normally as well
```
### Flow control
TegraScript has the following functions for flow control: `if()`, `else()`, `while()`, `return()`, `exit()`
- `if()` checks it's first arg, if it's 1, it runs the next {}, otherwise it skips over it
- `else()` checks if the last statement was an if, and if that if skipped, we run the next {}
- `while()` checks it's first arg, if it's 1, it runs the next {}, and jumps back to the while
- `return()` exits the current running function, if there is one
- `exit()` exits the script entirely
Let's try to build a loop that iterates 30 times, and printing even numbers
```
i = 0
while (i < 30){ # check if i is below 30, if so, run the code between the brackets
if (i % 2 == 0){ # is i dividable by 2?
println(i)
}
i = i + 1 # don't forget the + 1 of i! otherwise an infinite loop will haunt you
}
```
## Operators
### Integer, Integer
Operator | Output
|:-|:-|
`+` | sum of both integers
`-` | integer minus integer
`*` | multiplication of both integers
`/` | integer divided by integer
`%` | integer division remainder (modulo) of integer
`<` | 1 if left is smaller, otherwise 0
`>` | 1 if left is bigger, otherwise 0
`<=`| 1 if left is smaller or equal, otherwise 0
`>=`| 1 if left is bigger or equal, otherwise 0
`==`| 1 if left and right are equal, otherwise 0
`!=`| 1 if left and right are not equal, otherwise 0
`&&`| 1 if left and right are non 0, otherwise 0. Also if output is 0, disregards rest of statement
`\|\|`| 1 if left or right are non 0, otherwise 0. Also if output is 1, disregards rest of statement
`&` | Binary operator. ANDs both integers together
`\|` | Binary operator. ORs both integers together
`<<`| Binary operator. Bitshifts the left integer to the left by right integer's amount
`>>`| Binary operator. Bitshifts the left integer to the right by right integer's amount
### String, String
Operator | Output
|:-|:-|
`+` | Adds both strings together
`==`| 1 if strings are equal, otherwise 0
`-` | Removes the end of the first string if the second string matches the end of the first. (Example: `"abcdef" - "def"` is `"abc"`)
`/` | Splits the left string based on the right string. Returns a string array
### (Integer Array, Byte Array), Integer
Operator | Output
|:-|:-|
`+` | Adds the right integer into the left array
`-` | Removes right integer amount of entries from the left array
`:` | Removes right integer amount of entries from the beginning of the left array
### String, Integer
Operator | Output
|:-|:-|
`-` | Removes right integer amount of characters from the left string
`:` | Removes right integer amount of character from the beginning of the left string
### StringArray, String
Operator | Output
|:-|:-|
`+` | Adds a string to an array
### EmptyArray, (String, Int)
Operator | Output
|:-|:-|
`+` | Creates the array of the right type and puts the given value as the first entry
## Functions
### Flow control functions
Name | Description | OutType
|:-|:-|:-|
`if(int arg)` | Runs the next {} if arg is non zero | None
`else()` | Runs the next {} if the last statement was an if that skipped their {} | None
`while(int arg)` | Runs the next {} if arg is non zero. Jumps to the if after exiting the {} | None
`return()` | Breaks out of a function | None
`exit()` | Exits the current script | None
### Utilities
Name | Description | OutType
|:-|:-|:-|
`print(...)` | Prints all args provided to the screen. Can print Integer, String, IntegerArray. `\r` and `\n` is supported | None
`println(...)` | Same as `print(...)` but puts a newline at the end | None
`color(string color)`| Sets the print color. Supported inputs: `"RED"`, `"ORANGE"`, `"YELLOW"`, `"GREEN"`, `"BLUE"`, `"VIOLET"`, `"WHITE"` | None
`len(var arg1)` | Gets the length of a string or an array | Integer
`byte(IntegerArray arg1)`| Converts an integer array to a byte one | ByteArray
`bytesToStr(ByteArray arg1)`| Converts a byte array to a string | String
`printPos(int x, int y)` | Sets the printing position on screen. X/Y are in whole character sizes (16x16) | None
`clearscreen()` | Clears the screen full of your nonsense | None
`drawBox(int x1, int y1, int x2, int y2, int color)` | Draws a box from x1/y1 to x2/y2 with the color as color (raw: 0x00RRGGBB) | None
`wait(int ms)` | Waits for ms amount | None
`pause()` | Pauses until controller input is detected. Returns the controller input as raw u32 bitfield | Integer
`version()` | Returns an Integer array of the current TE version | IntegerArray
`menu(StringArray options, int startPos)` | Makes a menu with the cursor at startPos, with the provided options. Returns the current pos when a is pressed. B always returns 0 | Integer
`menu(StringArray options, int startPos, StringArray colors)` | Same as above, but the entries now get colors defined by the colors array. Uses the same colors as the `colors()` function | Integer
`menu(StringArray options, int startPos, StringArray colors, IntegerArray modifier)` | Same as above, but entries can be hidden or skipped defined by the modifier array. 0 for normal, 1 for skip, 2 for hide | Integer
Note about `pause()`. You need to work with raw bitfields. Have an example
```
# The most common controls
JoyY = 0x1
JoyX = 0x2
JoyB = 0x4
JoyA = 0x8
LeftJoyDown = 0x10000
LeftJoyUp = 0x20000
LeftJoyRight = 0x40000
LeftJoyLeft = 0x80000
if (pause() & JoyX){
println("X has been pressed!")
}
```
### FileSystem functions
Name | Description | OutType
|:-|:-|:-|
`fileRead(string path)` | Reads the file at the given path and returns it's contents in a byte array | ByteArray
`fileWrite(string path, ByteArray data)` | Writes data to the given path. Returns non zero on error | Integer
`fileExists(string path)` | Checks if a file or folder exists at the given path. 1 if yes, otherwise 0 | Integer
`fileMove(string src, string dst)` | Moves a file from src to dst. Returns non zero on error | Integer
`fileCopy(string src, string dst)` | Copies a file from src to dst. Returns non zero on error | Integer
`fileDel(string path)` | Deletes the file located at path. Returns non zero on error | Integer
`pathCombine(...)` | Needs 2+ string args as input. Combines them into a path. First entry must be the source folder Example: `pathCombine("sd:/", "tegraexplorer")` -> `"sd:/tegraexplorer"` | String
`pathEscFolder(string path)`| Escapes a folder path. Example: `pathEscFolder("sd:/tegraexplorer")` -> `"sd:/"` | String
`dirRead(string path)` | Reads a folder and returns a StringArray of filenames. Also creates an IntegerArray called `fileProperties` that is the same length as the filenames, and is non zero if the index of the filename is a folder | StringArray
`dirCopy(string src, string dst)`| Copies a folder from src to dst. Dst needs to be the containing folder of where you want src to go (`"sd:/tegraexplorer", "sd:/backup` -> `"sd:/backup/tegraexplorer"`). Returns non zero on error | Integer
`dirDel(string path)` | Deletes the dir located at path. Returns non zero on error | Integer
`mkdir(string path)` | Makes a directory at path | Integer
`launchPayload(string path)`| Launches the payload at the given path. Returns non zero on error | Integer
### Storage functions !! Dangerous
Name | Description | OutType
|:-|:-|:-|
`mmcConnect(string loc)` | Loc can be `"SYSMMC"` or `"EMUMMC"`. Returns non zero on error | Integer
`mmcMount(string loc)` | Loc can be `"PRODINFOF"`, `"SAFE"`, `"SYSTEM"` and `"USER"`. Mounts the filesystem to the prefix `bis:/`. Returns non zero on error | Integer
`mmcDump(string path, string target)`| Dumps target to path. target can be anything in the EMMC menu in TE. Returns non zero on error | Integer
`mmcRestore(string path, string target, int force)`| Restores path to target. target can be anything in the EMMC menu in TE. Force forces smaller files to flash anyway. Returns non zero on error | Integer
`ncaGetType(string path)`| Returns the type of nca that is in path. Make sure you provide an nca and nothing else! Useful for differentiating between Meta and Non-Meta Nca's | Integer
`saveSign(string path)` | Signs the (system) save at the given location. Make sure you provide a save and nothing else! Returns non zero on error | Integer
# Changelog
### 11/1/2021 @ 22:10 // Wow it isn't 2am for once!
For the release of TE 3.0.1, the following changes were made:
- A new type was added: Empty array type. You can get this type by running `var = []`
- Operators were added for StringArray, String and EmptyArray, (String, Integer)
- launchPayload was added as a function
- Menu now allows for an additional integer array to define hidden or skip flags
- A bug was fixed with <= and >= registering as < and >
- fileCopy's should not leave an ugly [ anymore
- A script now get executed at TE bootup if a script called `startup.te` is found on the root of the SD
### 5/1/2021 @ 1:32am // Sleep is still a lie
- Initial writeup on TScript v2

View file

@ -1,122 +1,122 @@
checkIfImportantSave = {
importantSaves = ["8000000000000120", "80000000000000d1", "8000000000000047"]
j = 0
important = 0
while (j < len(importantSaves)){
if (importantSaves[j] == save){
important = 1
}
j = j + 1
}
}
ver = version()
println("Tegrascript system wiper")
println("Running on TE ", ver)
println()
BTN_X = 0x2
if (_EMU) {
menuOptions = ["Exit", "Sysmmc", "Emummc"]
}
else() {
menuOptions = ["Exit", "Sysmmc"]
}
print("Wipe from: ")
res = menu(menuOptions, 0)
clearscreen()
if (res == 0){
exit()
}
if (res == 1){
println("Mounting Sysmmc")
mount = mmcConnect("SYSMMC")
}
if (res == 2){
println("Mounting Emummc")
mount = mmcConnect("EMUMMC")
}
if (mount){
println("Error connecting mmc!")
pause()
exit()
}
if (mmcMount("SYSTEM")) {
println("Failed to mount SYSTEM")
pause()
exit()
}
color("RED")
println("Are you sure you want to wipe everything?\nThis includes:\n- Saves\n- Game Data\n- All other data on the system\n\nUse this only as a last resort!")
color("YELLOW")
wait(10000)
println("Press X to continue\nPress any other button to exit")
start = pause() & BTN_X
if (!start){
color("WHITE")
exit()
}
color("WHITE")
println("Deleting SYSTEM saves")
files = dirRead("bis:/save")
i = 0
color("RED")
while (i < len(files)) {
if (!fileProperties[i]){ # checks if it's not a file
save = files[i]
checkIfImportantSave()
if (!important){
print("\rDeleting ", save)
res = fileDel(pathCombine("bis:/save", save))
if (res) {
println("\nFile deletion failed!")
pause()
exit()
}
}
}
i = i + 1
}
color("WHITE")
println("\n\nDeleting USER\n")
color("RED")
if (mmcMount("USER")){
println("Failed to mount USER")
pause()
exit()
}
toDel = ["Album", "Contents", "save", "saveMeta", "temp"]
i = 0
while (i < len(toDel)){
dirDel(pathCombine("bis:/", toDel[i]))
mkdir(pathCombine("bis:/", toDel[i]))
i = i + 1
}
mkdir("bis:/Contents/placehld")
mkdir("bis:/Contents/registered")
color("GREEN")
println("\n\nDone!")
checkIfImportantSave = {
importantSaves = ["8000000000000120", "80000000000000d1", "8000000000000047"]
j = 0
important = 0
while (j < len(importantSaves)){
if (importantSaves[j] == save){
important = 1
}
j = j + 1
}
}
ver = version()
println("Tegrascript system wiper")
println("Running on TE ", ver)
println()
BTN_X = 0x2
if (_EMU) {
menuOptions = ["Exit", "Sysmmc", "Emummc"]
}
else() {
menuOptions = ["Exit", "Sysmmc"]
}
print("Wipe from: ")
res = menu(menuOptions, 0)
clearscreen()
if (res == 0){
exit()
}
if (res == 1){
println("Mounting Sysmmc")
mount = mmcConnect("SYSMMC")
}
if (res == 2){
println("Mounting Emummc")
mount = mmcConnect("EMUMMC")
}
if (mount){
println("Error connecting mmc!")
pause()
exit()
}
if (mmcMount("SYSTEM")) {
println("Failed to mount SYSTEM")
pause()
exit()
}
color("RED")
println("Are you sure you want to wipe everything?\nThis includes:\n- Saves\n- Game Data\n- All other data on the system\n\nUse this only as a last resort!")
color("YELLOW")
wait(10000)
println("Press X to continue\nPress any other button to exit")
start = pause() & BTN_X
if (!start){
color("WHITE")
exit()
}
color("WHITE")
println("Deleting SYSTEM saves")
files = dirRead("bis:/save")
i = 0
color("RED")
while (i < len(files)) {
if (!fileProperties[i]){ # checks if it's not a file
save = files[i]
checkIfImportantSave()
if (!important){
print("\rDeleting ", save)
res = fileDel(pathCombine("bis:/save", save))
if (res) {
println("\nFile deletion failed!")
pause()
exit()
}
}
}
i = i + 1
}
color("WHITE")
println("\n\nDeleting USER\n")
color("RED")
if (mmcMount("USER")){
println("Failed to mount USER")
pause()
exit()
}
toDel = ["Album", "Contents", "save", "saveMeta", "temp"]
i = 0
while (i < len(toDel)){
dirDel(pathCombine("bis:/", toDel[i]))
mkdir(pathCombine("bis:/", toDel[i]))
i = i + 1
}
mkdir("bis:/Contents/placehld")
mkdir("bis:/Contents/registered")
color("GREEN")
println("\n\nDone!")
pause()