Port KVG to Rust & write docs

Also, this patch adds a fix that gets rid of cursor offset for VSCode Web (e.g: vscode.dev, GH Codespaces)

TEST=compile KVG on all architectures
This commit is contained in:
kxtzownsu 2024-12-20 00:30:58 -05:00 committed by kxtz smith
parent 8ac198052c
commit bfb70c6ec9
9 changed files with 340 additions and 14 deletions

5
.cargo/config.toml Normal file
View File

@ -0,0 +1,5 @@
[target.aarch64-unknown-linux-musl]
linker = "aarch64-linux-musl-gcc"
[target.armv7-unknown-linux-musleabihf]
linker = "rust-lld"

7
.gitignore vendored
View File

@ -1,5 +1,7 @@
# nya # nya
build/ build/
target/
Cargo.lock
# Prerequisites # Prerequisites
*.d *.d
@ -53,3 +55,8 @@ modules.order
Module.symvers Module.symvers
Mkfile.old Mkfile.old
dkms.conf dkms.conf
# Added by cargo
/target

10
.vscode/settings.json vendored Normal file
View File

@ -0,0 +1,10 @@
{
"files.associations": {
"stdio.h": "c"
},
// this setting isn't "unknown", its just for troubleshooting,
// but it fixes the vscode.dev cursor on ChromeOS.
"editor.disableMonospaceOptimizations": true
}

18
Cargo.toml Normal file
View File

@ -0,0 +1,18 @@
[package]
name = "kvg-kvs"
version = "0.1.0"
authors = ["Katelyn"]
edition = "2021"
[workspace]
[dependencies]
clap = { version = "4.0", features = ["derive"] }
[[bin]]
name = "KVG"
path = "src/KVG/main.rs"
[[bin]]
name = "KVS"
path = "src/KVS/main.rs"

View File

@ -1,4 +1,6 @@
CC := gcc CC ?= gcc
ARCH ?= x86_64
TOOLCHAIN ?= musl
SHELL ?= /bin/sh SHELL ?= /bin/sh
KVSFLIST := \ KVSFLIST := \
src/KVS/main.c \ src/KVS/main.c \
@ -13,10 +15,22 @@ CFLAGS := \
-Llib \ -Llib \
-static -static
ifeq ($(ARCH), aarch64)
CC := aarch64-linux-gnu-gcc
endif
ifeq ($(ARCH), armv7)
CC := armv7-linux-gnu-gcc
TOOLCHAIN := musleabihf
endif
TARGET = ${ARCH}-unknown-linux-${TOOLCHAIN}
all: clean build kvs kvg all: clean build kvs kvg
kvs: build build/bin/kvs kvs: build build/bin/kvs
kvg: build build/bin/kvg kvg: build build/bin/kvg
kvg-c: build build/bin/kvg-c
build: build:
$(shell mkdir -p build/bin) $(shell mkdir -p build/bin)
@ -25,12 +39,19 @@ build/bin/kvs: src/KVS/main.c
$(CC) $(KVSFLIST) -o build/bin/kvs $(CFLAGS) $(CC) $(KVSFLIST) -o build/bin/kvs $(CFLAGS)
chmod +rx build/bin/kvs chmod +rx build/bin/kvs
build/bin/kvg: src/KVG/main.c build/bin/kvg: src/KVG/main.rs
$(CC) src/KVG/main.c -o build/bin/kvg $(CFLAGS) cargo build --bin KVG --target=$(TARGET) --release
chmod +rx build/bin/kvg cp target/$(TARGET)/release/KVG build/bin/kvg
# The C version of KVS, not normally built.
# Also guaranteed to be out-of-date.
build/bin/kvg-c: src/KVG/main.c
$(CC) src/KVG/main.c -o build/bin/kvg-c $(CFLAGS)
chmod +rx build/bin/kvg-c
install: install:
cp -r build/* /usr/local/ cp -r build/* /usr/local/
clean: clean:
rm -rf build rm -rf build
rm -rf target

View File

@ -1,10 +1,10 @@
# Building KVS: # Building KVS:
### Dependencies ### Dependencies
You only need gcc & make! All static libs are inside `lib/` You only need gcc, make, and musl-tools! All static libs are inside `lib/`
``` ```
Debian-based systems: sudo apt install gcc make Debian-based systems: sudo apt install gcc make musl-tools
Arch-based systems: sudo pacman -S gcc make Arch-based systems: sudo pacman -S gcc make musl
Alpine-based systems: apk add gcc make Alpine-based systems: apk add gcc make
``` ```
@ -19,8 +19,8 @@ git clone https://github.com/kxtzownsu/KVS
cd KVS cd KVS
# Third, run two command to compile the KVS & KVG binary # Third, run two command to compile the KVS & KVG binary
make kvs # final binary is at ./build/kvs make kvs # final binary is at ./build/bin/kvs
make kvg # final binary is at ./build/kvg make kvg # final binary is at ./build/bin/kvg
# (OPTIONAL) Fourth, run the shim builder # (OPTIONAL) Fourth, run the shim builder
sudo make shim-builder sudo make shim-builder
@ -35,11 +35,11 @@ Notes: KVS **requires** KVG or else the shim will not build successfully
# Building KVG: # Building KVG:
### Dependencies ### Dependencies
Same as KVS, you only need `gcc` and `make` Same as KVS, you only need `gcc`, `make`, and `musl`
``` ```
Debian-based systems: sudo apt install gcc make Debian-based systems: sudo apt install gcc make musl-tools
Arch-based systems: sudo pacman -S gcc make Arch-based systems: sudo pacman -S gcc make musl
Alpine-based systems: apk add gcc make Alpine-based systems: apk add gcc make
``` ```
@ -50,5 +50,8 @@ git clone https://github.com/kxtzownsu/KVS # insiders, use KVS-private
cd KVS cd KVS
make kvg # final binary will be at ./build/kvg make kvg # final binary will be at ./build/bin/kvg
``` ```
# Cross-compiling
See [cross-compilation.md](cross-compilation.md).

67
docs/USAGE.md Normal file
View File

@ -0,0 +1,67 @@
# Using KVG (Kernel Version Generator)
Using KVG is very simple, all you need to know is if your device has v0 or v1.
This can be figured out very easily, if your device is oldui (white mode) recovery, then its v0,
if your device is newui (dark mode) recovery, then its v1.<br><br>
CLI Args:
```
$ ./kvg
USAGE: ./kvg <kernver> <optional flags>
e.g.: ./kvg 0x00010001 --raw --ver=0
--raw - prints the output as raw hex bytes
--ver=<0/1> - specifies the kernver struct version to use
--help - shows this message :3
KVG was created by kxtzownsu
(now written in Rust)
$
```
To use V0 recovery, pass `--ver=0` onto the END of the command, after putting the binary,
pass your flags, like this:
`./kvg 0x00010001 --ver=0 --raw`
NOT like this:
`./kvg --raw --ver=0 0x00010001`
Passing `--raw` will give you the raw hex output, instead of it printing like this `02 4c` it would print `\x2\x4c`.
# Using KVS (Kernel Version Switcher)
One thing to note, if you aren't using cr50-hammer or RMASmoke, then you must be **UNENROLLED**!
If you *are* using RMASmoke, make sure you're on Cr50 RW Version 0.5.229 or lower.
If you *are* using cr50-hammer, make sure <fill in later when cr50-hammer patch>.
If you *aren't* using RMASmoke or cr50-hammer, make sure to use the `tpm0 flash` method!
## Examples:
***Flashing via tpm0 flash***
```
KVS: Kernel Version Switcher (codename Maglev, bid: 2.0.0))
FW Version: Google_Grunt.11031.149.0
Kernel Version: 0x00010001
TPM: 2.0
FWMP: 0x1
GSC RW Version: 0.5.229
GSC Type: Cr50
-=-=-=-=-=-=-=-=-=-=-=-=-
1) Flash new kernver via /dev/tpm0 (REQ. UNENROLLED)
2) Flash new kernver via RMASmoke (REQ. CR50 VER 0.5.229 OR LOWER)
3) Make kernver index unwritable
4) Shell
5) Reboot
> 1
What kernver would you like to flash?
> 0x00010001
Does your device have lightmode (v0) or darkmode (v1) recovery? Please type either v0 or v1.
> v0
writing 13 bytes...
Finished! Press ENTER to return to main menu
```
<finish docs when RMASmoke & cr50-hammer release>

10
docs/cross-compilation.md Normal file
View File

@ -0,0 +1,10 @@
# Cross-compiling the KVS project
To cross-compile the KVS project, all you need to do is set `$ARCH` to a correct architecture,
and also install the cross-compilation libraries for your target arch.
Heres an example:
`ARCH=aarch64 make all`
All architectures:
`x86_64, armv7, aarch64`

185
src/KVG/main.rs Normal file
View File

@ -0,0 +1,185 @@
use std::env;
use std::process;
use std::io::{self, Write};
const VB2_SHA256_DIGEST_SIZE: usize = 32;
const MAX_HEX_LENGTH: usize = 10;
#[repr(C, packed)]
struct Vb2SecdataKernelV0 {
struct_version: u8,
uid: u32,
kernver: u32,
reserved: [u8; 3],
crc8: u8,
}
#[repr(C, packed)]
struct Vb2SecdataKernelV1 {
struct_version: u8,
struct_size: u8,
crc8: u8,
flags: u8,
kernver: u32,
ec_hash: [u8; VB2_SHA256_DIGEST_SIZE],
}
struct Vb2Context<'a> {
secdata_kernel: &'a [u8],
}
fn vb2_crc8(data: &[u8]) -> u8 {
let mut crc: u16 = 0;
for byte in data {
crc ^= (*byte as u16) << 8;
for _ in 0..8 {
if crc & 0x8000 != 0 {
crc ^= 0x1070 << 3;
}
crc <<= 1;
}
}
(crc >> 8) as u8
}
fn is_v0(context: &Vb2Context) -> bool {
let major_version = (context.secdata_kernel[0] & 0xf0) >> 4;
major_version == 0
}
fn secdata_kernel_crc(context: &Vb2Context) -> u8 {
if is_v0(context) {
let size = std::mem::size_of::<Vb2SecdataKernelV0>() - 1; // Exclude crc8
vb2_crc8(&context.secdata_kernel[0..size])
} else {
let struct_size = context.secdata_kernel[1] as usize;
let offset = 3; // Offset for flags in V1
vb2_crc8(&context.secdata_kernel[offset..struct_size])
}
}
fn is_valid_hex(input: &str) -> bool {
if input.len() > MAX_HEX_LENGTH {
return false;
}
if let Some(stripped) = input.strip_prefix("0x") {
stripped.chars().all(|c| c.is_digit(16))
} else {
false
}
}
fn sanitize_input(input: &str) -> String {
input.replace('\n', "\0")
}
fn convert_to_u32(input: &str) -> u32 {
u32::from_str_radix(input.trim_start_matches("0x"), 16).unwrap_or_else(|_| {
eprintln!(
"The entered kernver, {}, was not valid hexadecimal. Please try again.",
input
);
process::exit(1);
})
}
fn print_hex(data: &[u8]) {
for byte in data {
print!("{:02x} ", byte);
}
println!();
}
fn main() {
let args: Vec<String> = env::args().collect();
if args.len() < 2 || args.contains(&"--help".to_string()) {
println!(
"USAGE: {} <kernver> <optional flags>\n\
e.g.: {} 0x00010001 --raw\n\
--raw - prints the output as raw hex bytes\n\
--ver=<0/1> - specifies the kernver struct version to use\n\
--help - shows this message :3\n\
KVG was created by kxtzownsu\n\
(now written in Rust)",
args[0], args[0]
);
process::exit(0);
}
let kernver = sanitize_input(&args[1]);
if !is_valid_hex(&kernver) {
eprintln!(
"The entered kernver: {}, wasn't detected as valid hexadecimal (max 10 characters). Please try again.",
kernver
);
process::exit(1);
}
let kvarg = convert_to_u32(&kernver);
let version = args.iter().find(|arg| arg.starts_with("--ver=")).map_or("", |v| &v[6..]);
if version != "0" && version != "1" {
eprintln!(
"The entered struct version: {}, wasn't valid (see --help). Please try again.",
version
);
process::exit(1);
}
let mut buffer = Vec::new();
if version == "0" {
let mut secdata0 = Vb2SecdataKernelV0 {
struct_version: 0x02,
uid: 0x4752574c,
kernver: kvarg,
reserved: [0x00; 3],
crc8: 0,
};
let ctx = Vb2Context {
secdata_kernel: unsafe {
std::slice::from_raw_parts(
&secdata0 as *const _ as *const u8,
std::mem::size_of::<Vb2SecdataKernelV0>(),
)
},
};
secdata0.crc8 = secdata_kernel_crc(&ctx);
buffer.extend_from_slice(unsafe {
std::slice::from_raw_parts(
&secdata0 as *const _ as *const u8,
std::mem::size_of::<Vb2SecdataKernelV0>(),
)
});
} else if version == "1" {
let mut secdata1 = Vb2SecdataKernelV1 {
struct_version: 0x10,
struct_size: std::mem::size_of::<Vb2SecdataKernelV1>() as u8,
crc8: 0,
flags: 0x0,
kernver: kvarg,
ec_hash: [0; VB2_SHA256_DIGEST_SIZE],
};
let ctx = Vb2Context {
secdata_kernel: unsafe {
std::slice::from_raw_parts(
&secdata1 as *const _ as *const u8,
std::mem::size_of::<Vb2SecdataKernelV1>(),
)
},
};
secdata1.crc8 = secdata_kernel_crc(&ctx);
buffer.extend_from_slice(unsafe {
std::slice::from_raw_parts(
&secdata1 as *const _ as *const u8,
std::mem::size_of::<Vb2SecdataKernelV1>(),
)
});
}
if args.contains(&"--raw".to_string()) {
io::stdout().write_all(&buffer).unwrap();
} else {
print_hex(&buffer);
}
}