mirror of
https://github.com/suchmememanyskill/TegraScript.git
synced 2025-01-24 17:57:55 -03:00
start of docs of tsv3
This commit is contained in:
parent
9a0239b78f
commit
e887ef6e10
5 changed files with 551 additions and 343 deletions
402
README.md
402
README.md
|
@ -1,250 +1,208 @@
|
|||
# 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 veriable 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`
|
||||
|
||||
# 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 length of integer |
|
||||
| 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 |
|
||||
|
||||
- Initial writeup on TScript v2
|
||||
|
|
250
v2/README.md
Normal file
250
v2/README.md
Normal 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
|
|
@ -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()
|
Loading…
Add table
Reference in a new issue