Save cust config to localStorage
This commit is contained in:
1
Source/loaderConfigurator/.gitignore
vendored
1
Source/loaderConfigurator/.gitignore
vendored
@@ -1,3 +1,2 @@
|
||||
node_modules/
|
||||
dist/
|
||||
package-lock.json
|
||||
|
||||
6
Source/loaderConfigurator/build.sh
Executable file
6
Source/loaderConfigurator/build.sh
Executable file
@@ -0,0 +1,6 @@
|
||||
#!/bin/bash
|
||||
|
||||
[ -d "./dist" ] || mkdir ./dist
|
||||
cp -Rf ./src/*.html ./dist/
|
||||
tsc ./src/main.ts --outDir ./dist/ -lib es2015,dom -t es2015
|
||||
npx tailwindcss -i ./src/style.css -o ./dist/output.css
|
||||
22
Source/loaderConfigurator/dist/index.html
vendored
Normal file
22
Source/loaderConfigurator/dist/index.html
vendored
Normal file
@@ -0,0 +1,22 @@
|
||||
<!DOCTYPE html>
|
||||
<html class="dark">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<link href="./output.css" rel="stylesheet">
|
||||
</head>
|
||||
<body class="bg-slate-200 dark:bg-slate-900">
|
||||
<div class="flex justify-center">
|
||||
<div id="form" class="form-group p-6 rounded-lg shadow-lg max-w-md my-10 bg-slate-100 dark:bg-slate-800">
|
||||
<label for="file" class="form-entry">Upload loader.kip</label>
|
||||
<a href="https://github.com/KazushiMe/Switch-OC-Suite/releases/latest" target="_blank" class="link-primary">Get latest here</a>
|
||||
<input id="file" type="file" class="form-control form-entry">
|
||||
<div class="btn">
|
||||
<button id="load" class="hide btn btn-secondary">Load Default</button>
|
||||
<button id="save" class="hide btn btn-primary">Save</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
<script type="text/javascript" src="./main.js"></script>
|
||||
</html>
|
||||
237
Source/loaderConfigurator/dist/main.js
vendored
Normal file
237
Source/loaderConfigurator/dist/main.js
vendored
Normal file
@@ -0,0 +1,237 @@
|
||||
const CUST_REV = 2;
|
||||
var buffer;
|
||||
function FindMagicOffset(buffer) {
|
||||
let view = new DataView(buffer);
|
||||
for (let i = 0; i < view.byteLength; i += 4) {
|
||||
if (view.getUint32(i, true) == 0x54535543) { // "CUST"
|
||||
return i;
|
||||
}
|
||||
}
|
||||
throw new Error("Invalid loader.kip file");
|
||||
}
|
||||
function CustEntry(name, size, desc, defval, minmax = [0, 1000000], step = 1, extra_validator = null) {
|
||||
this.name = name;
|
||||
this.size = size;
|
||||
this.desc = desc;
|
||||
this.defval = defval;
|
||||
this.min = minmax[0];
|
||||
this.max = minmax[1];
|
||||
this.step = step;
|
||||
this.validator = extra_validator;
|
||||
this.value = null;
|
||||
this.offset = null;
|
||||
}
|
||||
function InitCustTable() {
|
||||
let cust = [
|
||||
new CustEntry("mtcConf", 2, "<b>DRAM Timing</b>\
|
||||
<li><b>0</b>: AUTO_ADJ_MARIKO_SAFE: Auto adjust timings for LPDDR4 ≤3733 Mbps specs, 8Gb density.</li>\
|
||||
<li><b>1</b>: AUTO_ADJ_MARIKO_4266: Auto adjust timings for LPDDR4X 4266 Mbps specs, 8Gb density.</li>\
|
||||
<li><b>2</b>: ENTIRE_TABLE_ERISTA: Not implemented.</li>\
|
||||
<li><b>3</b>: ENTIRE_TABLE_MARIKO: Not implemented.</li>", 0, [0, 3]),
|
||||
new CustEntry("marikoCpuMaxClock", 4, "<b>Mariko CPU Max Clock in kHz</b>\
|
||||
<li>System default: 1785000</li>\
|
||||
<li>≥ 2193000 will enable overvolting (> 1120 mV)</li>", 2397000, [1785000, 3000000], 100, (x) => { return (x % 100) == 0; }),
|
||||
new CustEntry("marikoCpuBoostClock", 4, "<b>Mariko CPU Boost Clock in kHz</b>\
|
||||
<li>System default: 1785000</li>\
|
||||
<li>Must not be higher than marikoCpuMaxClock</li>", 1785000, [1785000, 3000000], 100, (x) => { return (x % 100) == 0; }),
|
||||
new CustEntry("marikoCpuMaxVolt", 4, "<b>Mariko CPU Max Voltage in mV</b>\
|
||||
<li>System default: 1120</li>\
|
||||
<li>Acceptable range: 1100 ≤ x ≤ 1300</li>", 1220, [1100, 1300]),
|
||||
new CustEntry("marikoGpuMaxClock", 4, "<b>Mariko GPU Max Clock in kHz</b>\
|
||||
<li>System default: 921600</li>\
|
||||
<li>Tegra X1+ official maximum: 1267200</li>", 1305600, [768000, 1536000], 100, (x) => { return (x % 100) == 0; }),
|
||||
new CustEntry("marikoEmcMaxClock", 4, "<b>Mariko RAM Max Clock in kHz</b>\
|
||||
<li>Values should be > 1600000, and divided evenly by 3200.</li>\
|
||||
<li><b>WARNING:</b> RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM</li>", 1996800, [1612800, 2400000], 3200, (x) => { return (x % 3200) == 0; }),
|
||||
new CustEntry("eristaCpuOCEnable", 4, "<b>Erista CPU Enable Overclock</b>\
|
||||
<li>Not usable unless CPU cvb table is filled in</li>", 0, [0, 1]),
|
||||
new CustEntry("eristaCpuMaxVolt", 4, "<b>Erista CPU Max Voltage in mV</b>\
|
||||
<li>Acceptable range: 1100 ≤ x ≤ 1400</li>\
|
||||
<li>Not enabled by default</li>", 0, [0, 1400], 100, (x) => { return x >= 1100; }),
|
||||
new CustEntry("eristaEmcMaxClock", 4, "<b>Erista RAM Max Clock in kHz</b>\
|
||||
<li>Values should be > 1600000, and divided evenly by 3200.</li>\
|
||||
<li><b>WARNING:</b> RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM</li>", 1862400, [1600000, 2400000], 3200, (x) => { return (x % 3200) == 0; }),
|
||||
new CustEntry("eristaEmcVolt", 4, "<b>Erista RAM Voltage in uV</b>\
|
||||
<li>Acceptable range: 1100000 ≤ x ≤ 1250000, and it should be divided evenly by 12500.</li>\
|
||||
<li>Not enabled by default</li>", 0, [0, 1250000], 12500, (x) => { return (x % 12500) == 0 && x >= 1100000; }),
|
||||
];
|
||||
return cust;
|
||||
}
|
||||
function ValidateCust(cust) {
|
||||
for (let i of cust) {
|
||||
if (i.value == 0)
|
||||
continue;
|
||||
if (i.value < i.min || i.value > i.max) {
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error(`Expected range: ${i.min} ≤ ${i.name} ≤ ${i.max}, got ${i.value}`);
|
||||
}
|
||||
if (i.validator && !i.validator(i.value)) {
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error(`Invalid value: ${i.value}(${i.name})\nValidator: ${i.validator}`);
|
||||
}
|
||||
}
|
||||
}
|
||||
function SaveCust(cust, buffer) {
|
||||
let dict = Object.assign({}, ...cust.map((x) => ({ [x.name]: x })));
|
||||
let view = new DataView(buffer);
|
||||
let storage = {};
|
||||
for (let i of cust) {
|
||||
let id = i.name;
|
||||
i.value = document.getElementById(id).value;
|
||||
storage[i.name] = i.value;
|
||||
switch (i.size) {
|
||||
case 2:
|
||||
view.setUint16(i.offset, i.value, true);
|
||||
break;
|
||||
case 4:
|
||||
view.setUint32(i.offset, i.value, true);
|
||||
break;
|
||||
default:
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error("Unknown size at " + i);
|
||||
}
|
||||
}
|
||||
ValidateCust(cust);
|
||||
storage["custRev"] = CUST_REV;
|
||||
localStorage.setItem("last_saved", JSON.stringify(storage));
|
||||
let a = document.createElement("a");
|
||||
a.href = window.URL.createObjectURL(new Blob([buffer], { type: "application/octet-stream" }));
|
||||
a.download = "loader.kip";
|
||||
a.click();
|
||||
}
|
||||
function LastSaved() {
|
||||
let storage = localStorage.getItem("last_saved");
|
||||
if (!storage) {
|
||||
return false;
|
||||
}
|
||||
let sObj = JSON.parse(storage);
|
||||
if (!sObj["custRev"] || sObj["custRev"] != CUST_REV) {
|
||||
localStorage.removeItem("last_saved");
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
function LoadLastSaved() {
|
||||
if (LastSaved()) {
|
||||
let storage = localStorage.getItem("last_saved");
|
||||
let sObj = JSON.parse(storage);
|
||||
for (let key in sObj) {
|
||||
if (key == "custRev") {
|
||||
continue;
|
||||
}
|
||||
document.getElementById(key).value = sObj[key];
|
||||
}
|
||||
}
|
||||
}
|
||||
function LoadDefault(cust) {
|
||||
let dict = Object.assign({}, ...cust.map((x) => ({ [x.name]: x })));
|
||||
for (let i of cust) {
|
||||
let id = i.name;
|
||||
document.getElementById(id).value = i.defval;
|
||||
}
|
||||
}
|
||||
function UpdateHTMLForm(cust) {
|
||||
let dict = Object.assign({}, ...cust.map((x) => ({ [x.name]: x })));
|
||||
let form = document.getElementById("form");
|
||||
for (let i of cust) {
|
||||
let id = i.name;
|
||||
let input = document.getElementById(id);
|
||||
if (!input) {
|
||||
let div = document.createElement("div");
|
||||
div.classList.add("form-floating", "my-3");
|
||||
let label = document.createElement("label");
|
||||
label.setAttribute("for", id);
|
||||
label.innerHTML = id;
|
||||
label.classList.add("form-entry");
|
||||
input = document.createElement("input");
|
||||
input.classList.add("form-control", "form-entry");
|
||||
input.min = dict[i.name].min;
|
||||
input.max = dict[i.name].max;
|
||||
input.id = id;
|
||||
input.type = "number";
|
||||
input.step = dict[i.name].step;
|
||||
div.appendChild(input);
|
||||
div.appendChild(label);
|
||||
let desc = dict[i.name].desc;
|
||||
if (desc) {
|
||||
let tip = document.createElement("small");
|
||||
tip.innerHTML = desc;
|
||||
tip.classList.add("form-entry");
|
||||
tip.setAttribute("for", id);
|
||||
div.appendChild(tip);
|
||||
}
|
||||
form.appendChild(div);
|
||||
}
|
||||
input.value = dict[i.name].value;
|
||||
}
|
||||
let btn = document.getElementById("load");
|
||||
btn.classList.remove("hide");
|
||||
if (LastSaved()) {
|
||||
btn.innerHTML = "Load Last Saved";
|
||||
btn.addEventListener('click', () => {
|
||||
LoadLastSaved();
|
||||
});
|
||||
}
|
||||
else {
|
||||
btn.addEventListener('click', () => {
|
||||
LoadDefault(cust);
|
||||
});
|
||||
}
|
||||
btn = document.getElementById("save");
|
||||
btn.classList.remove("hide");
|
||||
btn.addEventListener('click', () => {
|
||||
try {
|
||||
SaveCust(cust, buffer);
|
||||
}
|
||||
catch (e) {
|
||||
console.log(e);
|
||||
alert(e);
|
||||
}
|
||||
});
|
||||
}
|
||||
function ParseCust(magicOffset, buffer) {
|
||||
let view = new DataView(buffer);
|
||||
let cust = InitCustTable();
|
||||
let offset = magicOffset + 4;
|
||||
let rev = view.getUint16(offset, true);
|
||||
if (rev != CUST_REV) {
|
||||
throw new Error("Unsupported custRev, expected: " + CUST_REV + ", got " + rev);
|
||||
}
|
||||
offset += 2;
|
||||
for (let i of cust) {
|
||||
i.offset = offset;
|
||||
switch (i.size) {
|
||||
case 2:
|
||||
i.value = view.getUint16(offset, true);
|
||||
break;
|
||||
case 4:
|
||||
i.value = view.getUint32(offset, true);
|
||||
break;
|
||||
default:
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error("Unknown size at " + i);
|
||||
}
|
||||
offset += i.size;
|
||||
}
|
||||
ValidateCust(cust);
|
||||
UpdateHTMLForm(cust);
|
||||
}
|
||||
const fileInput = document.getElementById("file");
|
||||
fileInput.addEventListener('change', (event) => {
|
||||
let reader = new FileReader();
|
||||
reader.readAsArrayBuffer(event.target.files[0]);
|
||||
reader.onloadend = (progEvent) => {
|
||||
if (progEvent.target.readyState === FileReader.DONE) {
|
||||
buffer = progEvent.target.result;
|
||||
try {
|
||||
let offset = FindMagicOffset(buffer);
|
||||
ParseCust(offset, buffer);
|
||||
}
|
||||
catch (e) {
|
||||
console.log(e);
|
||||
alert(e);
|
||||
fileInput.value = "";
|
||||
}
|
||||
}
|
||||
};
|
||||
});
|
||||
@@ -660,24 +660,12 @@ video {
|
||||
opacity: 0.65;
|
||||
}
|
||||
|
||||
.btn-check[disabled] + .\!btn {
|
||||
pointer-events: none !important;
|
||||
filter: none !important;
|
||||
opacity: 0.65 !important;
|
||||
}
|
||||
|
||||
.btn-check:disabled + .btn {
|
||||
pointer-events: none;
|
||||
filter: none;
|
||||
opacity: 0.65;
|
||||
}
|
||||
|
||||
.btn-check:disabled + .\!btn {
|
||||
pointer-events: none !important;
|
||||
filter: none !important;
|
||||
opacity: 0.65 !important;
|
||||
}
|
||||
|
||||
.form-floating {
|
||||
position: relative;
|
||||
}
|
||||
@@ -766,19 +754,10 @@ video {
|
||||
z-index: 2;
|
||||
}
|
||||
|
||||
.input-group .\!btn {
|
||||
position: relative !important;
|
||||
z-index: 2 !important;
|
||||
}
|
||||
|
||||
.input-group .btn:focus {
|
||||
z-index: 3;
|
||||
}
|
||||
|
||||
.input-group .\!btn:focus {
|
||||
z-index: 3 !important;
|
||||
}
|
||||
|
||||
.input-group:not(.has-validation) > .dropdown-toggle:nth-last-child(n+3) {
|
||||
border-top-right-radius: 0;
|
||||
border-bottom-right-radius: 0;
|
||||
@@ -957,85 +936,43 @@ textarea.form-control.is-invalid {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.btn-check:focus + .\!btn {
|
||||
outline: 0 !important;
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.btn:focus {
|
||||
outline: 0;
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.\!btn:focus {
|
||||
outline: 0 !important;
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.btn-check:checked + .btn {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.btn-check:checked + .\!btn {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.btn-check:active + .btn {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.btn-check:active + .\!btn {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.btn:active {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.\!btn:active {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.btn.active {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.\!btn.active {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.btn-check:checked + .btn:focus {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.btn-check:checked + .\!btn:focus {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.btn-check:active + .btn:focus {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.btn-check:active + .\!btn:focus {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.btn:active:focus {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.\!btn:active:focus {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.btn.active:focus {
|
||||
box-shadow: none;
|
||||
}
|
||||
|
||||
.\!btn.active:focus {
|
||||
box-shadow: none !important;
|
||||
}
|
||||
|
||||
.fade {
|
||||
transition: opacity 0.15s linear;
|
||||
}
|
||||
@@ -1780,6 +1717,18 @@ textarea.form-control.is-invalid {
|
||||
transform: none;
|
||||
}
|
||||
|
||||
.link-primary {
|
||||
color: #0d6efd;
|
||||
}
|
||||
|
||||
.link-primary:hover {
|
||||
color: #0a58ca;
|
||||
}
|
||||
|
||||
.link-primary:focus {
|
||||
color: #0a58ca;
|
||||
}
|
||||
|
||||
.sticky-top {
|
||||
position: sticky;
|
||||
top: 0;
|
||||
@@ -1898,10 +1847,6 @@ textarea.form-control.is-invalid {
|
||||
background-image: radial-gradient(circle, rgba(255, 255, 255, 0.2) 0, rgba(255, 255, 255, 0.3) 40%, rgba(255, 255, 255, 0.4) 50%, rgba(255, 255, 255, 0.5) 60%, rgba(255, 255, 255, 0) 70%);
|
||||
}
|
||||
|
||||
.\!btn .ripple-wave {
|
||||
background-image: radial-gradient(circle, rgba(255, 255, 255, 0.2) 0, rgba(255, 255, 255, 0.3) 40%, rgba(255, 255, 255, 0.4) 50%, rgba(255, 255, 255, 0.5) 60%, rgba(255, 255, 255, 0) 70%) !important;
|
||||
}
|
||||
|
||||
.ripple-surface-primary .ripple-wave {
|
||||
background-image: radial-gradient(circle, rgba(18, 102, 241, 0.2) 0, rgba(18, 102, 241, 0.3) 40%, rgba(18, 102, 241, 0.4) 50%, rgba(18, 102, 241, 0.5) 60%, rgba(18, 102, 241, 0) 70%);
|
||||
}
|
||||
@@ -3785,23 +3730,20 @@ textarea.form-control.is-invalid {
|
||||
position: sticky;
|
||||
}
|
||||
|
||||
.my-10 {
|
||||
margin-top: 2.5rem;
|
||||
margin-bottom: 2.5rem;
|
||||
}
|
||||
|
||||
.my-3 {
|
||||
margin-top: 0.75rem;
|
||||
margin-bottom: 0.75rem;
|
||||
}
|
||||
|
||||
.mb-2 {
|
||||
margin-bottom: 0.5rem;
|
||||
}
|
||||
|
||||
.block {
|
||||
display: block;
|
||||
}
|
||||
|
||||
.inline-block {
|
||||
display: inline-block;
|
||||
}
|
||||
|
||||
.inline {
|
||||
display: inline;
|
||||
}
|
||||
@@ -3826,8 +3768,8 @@ textarea.form-control.is-invalid {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.max-w-sm {
|
||||
max-width: 24rem;
|
||||
.max-w-md {
|
||||
max-width: 28rem;
|
||||
}
|
||||
|
||||
.transform {
|
||||
@@ -3866,30 +3808,24 @@ textarea.form-control.is-invalid {
|
||||
justify-content: space-evenly;
|
||||
}
|
||||
|
||||
.space-x-2 > :not([hidden]) ~ :not([hidden]) {
|
||||
--tw-space-x-reverse: 0;
|
||||
margin-right: calc(0.5rem * var(--tw-space-x-reverse));
|
||||
margin-left: calc(0.5rem * calc(1 - var(--tw-space-x-reverse)));
|
||||
}
|
||||
|
||||
.rounded-lg {
|
||||
border-radius: 0.5rem;
|
||||
}
|
||||
|
||||
.bg-white {
|
||||
.bg-slate-200 {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(255 255 255 / var(--tw-bg-opacity));
|
||||
background-color: rgb(226 232 240 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
.bg-slate-100 {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(241 245 249 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
.p-6 {
|
||||
padding: 1.5rem;
|
||||
}
|
||||
|
||||
.text-gray-700 {
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(55 65 81 / var(--tw-text-opacity));
|
||||
}
|
||||
|
||||
.shadow-lg {
|
||||
--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / 0.1), 0 4px 6px -4px rgb(0 0 0 / 0.1);
|
||||
--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);
|
||||
@@ -3913,7 +3849,14 @@ label.form-entry {
|
||||
margin-bottom: 0.5rem;
|
||||
display: inline-block;
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(55 65 81 / var(--tw-text-opacity));
|
||||
color: rgb(51 65 85 / var(--tw-text-opacity));
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
label.form-entry {
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(226 232 240 / var(--tw-text-opacity));
|
||||
}
|
||||
}
|
||||
|
||||
input.form-entry {
|
||||
@@ -3924,9 +3867,9 @@ input.form-entry {
|
||||
border-width: 1px;
|
||||
border-style: solid;
|
||||
--tw-border-opacity: 1;
|
||||
border-color: rgb(209 213 219 / var(--tw-border-opacity));
|
||||
border-color: rgb(203 213 225 / var(--tw-border-opacity));
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(255 255 255 / var(--tw-bg-opacity));
|
||||
background-color: rgb(241 245 249 / var(--tw-bg-opacity));
|
||||
background-clip: padding-box;
|
||||
padding-left: 0.75rem;
|
||||
padding-right: 0.75rem;
|
||||
@@ -3936,7 +3879,7 @@ input.form-entry {
|
||||
line-height: 1.5rem;
|
||||
font-weight: 400;
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(55 65 81 / var(--tw-text-opacity));
|
||||
color: rgb(71 85 105 / var(--tw-text-opacity));
|
||||
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke, opacity, box-shadow, transform, filter, -webkit-backdrop-filter;
|
||||
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke, opacity, box-shadow, transform, filter, backdrop-filter;
|
||||
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke, opacity, box-shadow, transform, filter, backdrop-filter, -webkit-backdrop-filter;
|
||||
@@ -3948,13 +3891,33 @@ input.form-entry:focus {
|
||||
--tw-border-opacity: 1;
|
||||
border-color: rgb(37 99 235 / var(--tw-border-opacity));
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(255 255 255 / var(--tw-bg-opacity));
|
||||
background-color: rgb(241 245 249 / var(--tw-bg-opacity));
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(55 65 81 / var(--tw-text-opacity));
|
||||
color: rgb(51 65 85 / var(--tw-text-opacity));
|
||||
outline: 2px solid transparent;
|
||||
outline-offset: 2px;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
input.form-entry {
|
||||
--tw-border-opacity: 1;
|
||||
border-color: rgb(75 85 99 / var(--tw-border-opacity));
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(15 23 42 / var(--tw-bg-opacity));
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(203 213 225 / var(--tw-text-opacity));
|
||||
}
|
||||
|
||||
input.form-entry:focus {
|
||||
--tw-border-opacity: 1;
|
||||
border-color: rgb(147 197 253 / var(--tw-border-opacity));
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(30 41 59 / var(--tw-bg-opacity));
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(226 232 240 / var(--tw-text-opacity));
|
||||
}
|
||||
}
|
||||
|
||||
small.form-entry {
|
||||
margin-top: 0.25rem;
|
||||
display: block;
|
||||
@@ -3965,14 +3928,48 @@ small.form-entry {
|
||||
font-size: 0.75rem;
|
||||
line-height: 1rem;
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(75 85 99 / var(--tw-text-opacity));
|
||||
color: rgb(71 85 105 / var(--tw-text-opacity));
|
||||
}
|
||||
|
||||
button.save-btn {
|
||||
@media (prefers-color-scheme: dark) {
|
||||
small.form-entry {
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(148 163 184 / var(--tw-text-opacity));
|
||||
}
|
||||
}
|
||||
|
||||
a.link-primary {
|
||||
margin-bottom: 1rem;
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(59 130 246 / var(--tw-text-opacity));
|
||||
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke, opacity, box-shadow, transform, filter, -webkit-backdrop-filter;
|
||||
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke, opacity, box-shadow, transform, filter, backdrop-filter;
|
||||
transition-property: color, background-color, border-color, text-decoration-color, fill, stroke, opacity, box-shadow, transform, filter, backdrop-filter, -webkit-backdrop-filter;
|
||||
transition-duration: 300ms;
|
||||
transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1);
|
||||
}
|
||||
|
||||
a.link-primary:hover {
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(37 99 235 / var(--tw-text-opacity));
|
||||
}
|
||||
|
||||
a.link-primary:active {
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(29 78 216 / var(--tw-text-opacity));
|
||||
}
|
||||
|
||||
div.btn {
|
||||
margin-top: 0.75rem;
|
||||
display: grid;
|
||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
||||
-moz-column-gap: 1.5rem;
|
||||
column-gap: 1.5rem;
|
||||
}
|
||||
|
||||
button.btn {
|
||||
display: inline-block;
|
||||
border-radius: 0.25rem;
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(37 99 235 / var(--tw-bg-opacity));
|
||||
padding-left: 1.5rem;
|
||||
padding-right: 1.5rem;
|
||||
padding-top: 0.625rem;
|
||||
@@ -3983,7 +3980,7 @@ button.save-btn {
|
||||
text-transform: uppercase;
|
||||
line-height: 1.25;
|
||||
--tw-text-opacity: 1;
|
||||
color: rgb(255 255 255 / var(--tw-text-opacity));
|
||||
color: rgb(241 245 249 / var(--tw-text-opacity));
|
||||
--tw-shadow: 0 4px 6px -1px rgb(0 0 0 / 0.1), 0 2px 4px -2px rgb(0 0 0 / 0.1);
|
||||
--tw-shadow-colored: 0 4px 6px -1px var(--tw-shadow-color), 0 2px 4px -2px var(--tw-shadow-color);
|
||||
box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);
|
||||
@@ -3994,17 +3991,13 @@ button.save-btn {
|
||||
transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1);
|
||||
}
|
||||
|
||||
button.save-btn:hover {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(29 78 216 / var(--tw-bg-opacity));
|
||||
button.btn:hover {
|
||||
--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / 0.1), 0 4px 6px -4px rgb(0 0 0 / 0.1);
|
||||
--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);
|
||||
box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);
|
||||
}
|
||||
|
||||
button.save-btn:focus {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(29 78 216 / var(--tw-bg-opacity));
|
||||
button.btn:focus {
|
||||
--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / 0.1), 0 4px 6px -4px rgb(0 0 0 / 0.1);
|
||||
--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);
|
||||
box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);
|
||||
@@ -4015,10 +4008,64 @@ button.save-btn:focus {
|
||||
box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow, 0 0 #0000);
|
||||
}
|
||||
|
||||
button.save-btn:active {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(30 64 175 / var(--tw-bg-opacity));
|
||||
button.btn:active {
|
||||
--tw-shadow: 0 10px 15px -3px rgb(0 0 0 / 0.1), 0 4px 6px -4px rgb(0 0 0 / 0.1);
|
||||
--tw-shadow-colored: 0 10px 15px -3px var(--tw-shadow-color), 0 4px 6px -4px var(--tw-shadow-color);
|
||||
box-shadow: var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);
|
||||
}
|
||||
}
|
||||
|
||||
button.btn-primary {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(37 99 235 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
button.btn-primary:hover {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(29 78 216 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
button.btn-primary:focus {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(29 78 216 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
button.btn-primary:active {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(30 58 138 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
button.btn-secondary {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(71 85 105 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
button.btn-secondary:hover {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(51 65 85 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
button.btn-secondary:focus {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(51 65 85 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
button.btn-secondary:active {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(15 23 42 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
.hide {
|
||||
display: none !important;
|
||||
}
|
||||
|
||||
@media (prefers-color-scheme: dark) {
|
||||
.dark\:bg-slate-900 {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(15 23 42 / var(--tw-bg-opacity));
|
||||
}
|
||||
|
||||
.dark\:bg-slate-800 {
|
||||
--tw-bg-opacity: 1;
|
||||
background-color: rgb(30 41 59 / var(--tw-bg-opacity));
|
||||
}
|
||||
}
|
||||
173
Source/loaderConfigurator/ldr_config.py
Executable file
173
Source/loaderConfigurator/ldr_config.py
Executable file
@@ -0,0 +1,173 @@
|
||||
#!python3
|
||||
cust_conf = {
|
||||
# DRAM Timing:
|
||||
# 0: AUTO_ADJ_MARIKO_SAFE: Auto adjust timings for LPDDR4 ≤3733 Mbps specs, 8Gb density (Default).
|
||||
# 1: AUTO_ADJ_MARIKO_4266: Auto adjust timings for LPDDR4X 4266 Mbps specs, 8Gb density.
|
||||
# 2: ENTIRE_TABLE_ERISTA:
|
||||
# 3: ENTIRE_TABLE_MARIKO: Replace the entire max mtc table with customized one (provided by user).
|
||||
"mtcConf": 0,
|
||||
# Mariko CPU:
|
||||
# - Max Clock in kHz:
|
||||
# Default: 1785000
|
||||
# >= 2193000 will enable overvolting (> 1120 mV)
|
||||
# - Boost Clock in kHz:
|
||||
# Default: 1785000
|
||||
# Boost clock will be applied when applications request higher CPU frequency for quicker loading.
|
||||
# - Max Voltage in mV:
|
||||
# Default voltage: 1120
|
||||
# Haven't tested anything higher than 1220.
|
||||
"marikoCpuMaxClock": 2397000,
|
||||
"marikoCpuBoostClock": 1785000,
|
||||
"marikoCpuMaxVolt": 1220,
|
||||
# Mariko GPU:
|
||||
# - Max Clock in kHz:
|
||||
# Default: 921600
|
||||
# NVIDIA Maximum: 1267200
|
||||
"marikoGpuMaxClock": 1305600,
|
||||
# Mariko EMC:
|
||||
# - RAM Clock in kHz:
|
||||
# Values should be > 1600000, and divided evenly by 9600.
|
||||
# [WARNING]
|
||||
# RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM:
|
||||
# - Graphical glitches
|
||||
# - System instabilities
|
||||
# - NAND corruption
|
||||
# Timings from auto-adjustment have been tested safe for up to 1996.8 MHz for all DRAM chips.
|
||||
"marikoEmcMaxClock": 1996800,
|
||||
# Erista CPU:
|
||||
# Not tested and not enabled by default.
|
||||
# - Enable Overclock
|
||||
# Require modificaitions towards NewCpuTables!
|
||||
# - Max Voltage in mV
|
||||
"eristaCpuOCEnable": 0,
|
||||
"eristaCpuMaxVolt": 0,
|
||||
# Erista EMC:
|
||||
# - RAM Clock in kHz
|
||||
# [WARNING]
|
||||
# RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM:
|
||||
# - Graphical glitches
|
||||
# - System instabilities
|
||||
# - NAND corruption
|
||||
# - RAM Voltage in uV
|
||||
# Range: 600'000 to 1250'000 uV
|
||||
# Value should be divided evenly by 12'500
|
||||
# Default(HOS): 1125'000
|
||||
# Not enabled by default.
|
||||
"eristaEmcMaxClock": 1862400,
|
||||
"eristaEmcVolt": 0
|
||||
}
|
||||
|
||||
cust_range = {
|
||||
"mtcConf": (0, 3),
|
||||
"marikoCpuMaxClock": (1785000, 3000000),
|
||||
"marikoCpuBoostClock": (1785000, 3000000),
|
||||
"marikoCpuMaxVolt": (1100, 1300),
|
||||
"marikoGpuMaxClock": (768000, 1536000),
|
||||
"marikoEmcMaxClock": (1612800, 2400000),
|
||||
"eristaCpuMaxVolt": (1100, 1400),
|
||||
"eristaEmcMaxClock": (1600000, 2400000),
|
||||
"eristaEmcVolt": (1100000, 1250000)
|
||||
}
|
||||
|
||||
|
||||
import struct
|
||||
import argparse
|
||||
|
||||
cust_rev = 2
|
||||
cust_head = ["cust", "custRev"]
|
||||
cust_body = ["mtcConf",
|
||||
"marikoCpuMaxClock", "marikoCpuBoostClock", "marikoCpuMaxVolt", "marikoGpuMaxClock", "marikoEmcMaxClock",
|
||||
"eristaCpuOCEnable", "eristaCpuMaxVolt", "eristaEmcMaxClock", "eristaEmcVolt"]
|
||||
cust_key = [*cust_head, *cust_body]
|
||||
cust_val_num = len(cust_conf) - 1
|
||||
cust_fmt = '<4s2H' + str(cust_val_num) + 'I'
|
||||
cust_head_fmt = '<4s1H'
|
||||
cust_body_fmt = '<1H' + str(cust_val_num) + 'I'
|
||||
|
||||
parser = argparse.ArgumentParser(description='Loader Configurator v'+str(cust_rev))
|
||||
parser.add_argument("file", help="Path of loader.kip")
|
||||
parser.add_argument("--save", "-s", action="store_true", help="Save configuration to loader.kip")
|
||||
parser.add_argument("--ignore", action="store_true", help="Ignore range safety check")
|
||||
args = parser.parse_args()
|
||||
|
||||
def KIPCustParse(file_loc, conf_print=True) -> (int, dict):
|
||||
with open(file_loc, "rb") as file:
|
||||
header_str = b'KIP1Loader'
|
||||
header = file.read(len(header_str))
|
||||
file.seek(0)
|
||||
cust_magic = b'CUST'
|
||||
cust_pos = file.read().find(cust_magic)
|
||||
|
||||
if header != header_str or cust_pos == -1:
|
||||
raise Exception("\n Invalid kip file!")
|
||||
|
||||
file.seek(cust_pos)
|
||||
cust_size = struct.calcsize(cust_fmt)
|
||||
cust_buf = file.read(cust_size)
|
||||
cust_val = struct.unpack(cust_fmt, cust_buf)
|
||||
cust_dict = dict(zip(cust_key, cust_val))
|
||||
|
||||
if cust_dict['custRev'] != cust_rev:
|
||||
raise Exception(f"\n custRev does NOT match, expected: {cust_rev}, got: {cust_dict['custRev']}!")
|
||||
|
||||
[cust_dict.pop(key) for key in cust_head]
|
||||
|
||||
if conf_print:
|
||||
print("Configuration from file")
|
||||
[print(f"- {i:20s} : {cust_dict[i]:8d}") for i in cust_dict]
|
||||
|
||||
return (cust_pos, cust_dict)
|
||||
|
||||
|
||||
def CustRangeCheck(cust):
|
||||
range_error_str = ""
|
||||
for i in cust_range:
|
||||
val = int(cust[i])
|
||||
if val and (val < cust_range[i][0] or val > cust_range[i][1]) :
|
||||
range_error_str += f"\n- {i:20s} = {val:8d}, Expected range: {[*cust_range[i]]}"
|
||||
|
||||
if range_error_str:
|
||||
raise ValueError(range_error_str)
|
||||
|
||||
|
||||
def KIPCustSave(file_loc, cust_pos, cust_dict, range_check=True, cust_to_save={}):
|
||||
missing = set(cust_body) - set(cust_to_save.keys())
|
||||
if missing:
|
||||
missing_str = "\n Invalid cust! Missing: "
|
||||
for i in missing:
|
||||
missing_str += f"\n- {i}"
|
||||
raise Exception(missing_str)
|
||||
|
||||
if range_check:
|
||||
CustRangeCheck(cust_to_save)
|
||||
|
||||
diff_count = 0
|
||||
for i in cust_body:
|
||||
diff_str = ""
|
||||
if cust_dict[i] != cust_conf[i]:
|
||||
diff_str = f"-> {cust_conf[i]:8d}"
|
||||
diff_count += 1
|
||||
print(f"- {i:20s} : {cust_dict[i]:8d} {diff_str}")
|
||||
|
||||
if not diff_count:
|
||||
print("Cust is identical, abort saving!")
|
||||
return
|
||||
|
||||
with open(file_loc, "rb+") as file:
|
||||
cust_bin = struct.pack(cust_body_fmt, *[cust_to_save[i] for i in cust_body])
|
||||
file.seek(cust_pos + struct.calcsize(cust_head_fmt))
|
||||
file.write(cust_bin)
|
||||
|
||||
print("Done!")
|
||||
|
||||
|
||||
def main(file_loc, ignore=False, save=False):
|
||||
(cust_pos, cust_dict) = KIPCustParse(file_loc, conf_print=(not save))
|
||||
|
||||
if save:
|
||||
print("Saving new configuration...")
|
||||
KIPCustSave(file_loc, cust_pos, cust_dict, range_check=(not ignore), cust_to_save=cust_conf)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main(args.file, args.ignore, args.save)
|
||||
@@ -1,292 +1,22 @@
|
||||
<html>
|
||||
<!DOCTYPE html>
|
||||
<html class="dark">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<link href="./output.css" rel="stylesheet">
|
||||
</head>
|
||||
<body>
|
||||
<body class="bg-slate-200 dark:bg-slate-900">
|
||||
<div class="flex justify-center">
|
||||
<div id="form" class="form-group p-6 rounded-lg shadow-lg bg-white max-w-sm">
|
||||
<label for="file" class="inline-block mb-2 text-gray-700">Upload loader.kip</label>
|
||||
<div id="form" class="form-group p-6 rounded-lg shadow-lg max-w-md my-10 bg-slate-100 dark:bg-slate-800">
|
||||
<label for="file" class="form-entry">Upload loader.kip</label>
|
||||
<a href="https://github.com/KazushiMe/Switch-OC-Suite/releases/latest" target="_blank" class="link-primary">Get latest here</a>
|
||||
<input id="file" type="file" class="form-control form-entry">
|
||||
<div class="btn">
|
||||
<button id="load" class="hide btn btn-secondary">Load Default</button>
|
||||
<button id="save" class="hide btn btn-primary">Save</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
<script type="text/javascript">
|
||||
var buffer;
|
||||
|
||||
function FindMagicOffset(buffer) {
|
||||
let view = new DataView(buffer);
|
||||
for (let i = 0; i < view.byteLength; i += 4) {
|
||||
if (view.getUint32(i, true) == 0x54535543) { // "CUST"
|
||||
return i;
|
||||
}
|
||||
}
|
||||
throw new Error("Invalid loader.kip file");
|
||||
}
|
||||
|
||||
function CustEntry(name, size, desc, minmax = [0, 1_000_000], step = 1, extra_validator = undefined) {
|
||||
this.name = name;
|
||||
this.size = size;
|
||||
this.value = null;
|
||||
this.offset = null;
|
||||
this.desc = desc;
|
||||
this.min = minmax[0];
|
||||
this.max = minmax[1];
|
||||
this.step = step;
|
||||
this.validator = extra_validator;
|
||||
}
|
||||
|
||||
function InitCustTable() {
|
||||
let cust = [
|
||||
new CustEntry(
|
||||
"custRev",
|
||||
2,
|
||||
"Cust Version, must be 2",
|
||||
[2, 2],
|
||||
),
|
||||
new CustEntry(
|
||||
"mtcConf",
|
||||
2,
|
||||
"<b>DRAM Timing</b>\
|
||||
<li><b>0</b>: AUTO_ADJ_MARIKO_SAFE (Default): Auto adjust timings for LPDDR4 ≤3733 Mbps specs, 8Gb density.</li>\
|
||||
<li><b>1</b>: AUTO_ADJ_MARIKO_4266: Auto adjust timings for LPDDR4X 4266 Mbps specs, 8Gb density.</li>\
|
||||
<li><b>2</b>: ENTIRE_TABLE_ERISTA: Not implemented.</li>\
|
||||
<li><b>3</b>: ENTIRE_TABLE_MARIKO: Not implemented.</li>",
|
||||
[0, 3],
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoCpuMaxClock",
|
||||
4,
|
||||
"<b>Mariko CPU Max Clock in kHz</b>\
|
||||
<li>System default: 1785000</li>\
|
||||
<li>≥ 2193000 will enable overvolting (> 1120 mV)</li>",
|
||||
[1785_000, 3000_000],
|
||||
100,
|
||||
(x) => { return (x % 100) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoCpuBoostClock",
|
||||
4,
|
||||
"<b>Mariko CPU Boost Clock in kHz</b>\
|
||||
<li>System default: 1785000</li>\
|
||||
<li>Must not be higher than marikoCpuMaxClock</li>",
|
||||
[1785_000, 3000_000],
|
||||
100,
|
||||
(x) => { return (x % 100) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoCpuMaxVolt",
|
||||
4,
|
||||
"<b>Mariko CPU Max Voltage in mV</b>\
|
||||
<li>System default: 1120</li>\
|
||||
<li>Acceptable range: 1100 ≤ x ≤ 1300</li>",
|
||||
[1100, 1300],
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoGpuMaxClock",
|
||||
4,
|
||||
"<b>Mariko GPU Max Clock in kHz</b>\
|
||||
<li>System default: 921600</li>\
|
||||
<li>Tegra X1+ official maximum: 1267200</li>",
|
||||
[768_000, 1536_000],
|
||||
100,
|
||||
(x) => { return (x % 100) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoEmcMaxClock",
|
||||
4,
|
||||
"<b>Mariko RAM Max Clock in kHz</b>\
|
||||
<li>Values should be > 1600000, and divided evenly by 3200.</li>\
|
||||
<li><b>WARNING:</b> RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM</li>",
|
||||
[1612_800, 2400_000],
|
||||
3200,
|
||||
(x) => { return (x % 3200) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"eristaCpuOCEnable",
|
||||
4,
|
||||
"<b>Erista CPU Enable Overclock</b>\
|
||||
<li>Not usable unless CPU cvb table is filled in</li>",
|
||||
[0, 1]
|
||||
),
|
||||
new CustEntry(
|
||||
"eristaCpuMaxVolt",
|
||||
4,
|
||||
"<b>Erista CPU Max Voltage in mV</b>\
|
||||
<li>Acceptable range: 1100 ≤ x ≤ 1400</li>\
|
||||
<li>Not enabled by default</li>",
|
||||
[0, 1400],
|
||||
100,
|
||||
(x) => { return x >= 1100; }
|
||||
),
|
||||
new CustEntry(
|
||||
"eristaEmcMaxClock",
|
||||
4,
|
||||
"<b>Erista RAM Max Clock in kHz</b>\
|
||||
<li>Values should be > 1600000, and divided evenly by 3200.</li>\
|
||||
<li><b>WARNING:</b> RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM</li>",
|
||||
[1600_000, 2400_000],
|
||||
3200,
|
||||
(x) => { return (x % 3200) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"eristaEmcVolt",
|
||||
4,
|
||||
"<b>Erista RAM Voltage in uV</b>\
|
||||
<li>Acceptable range: 1100000 ≤ x ≤ 1250000, and it should be divided evenly by 12500.</li>\
|
||||
<li>Not enabled by default</li>",
|
||||
[0, 1250_000],
|
||||
12500,
|
||||
(x) => { return (x % 12500) == 0 && x >= 1100_000; }
|
||||
),
|
||||
];
|
||||
return cust;
|
||||
}
|
||||
|
||||
function ValidateCust(cust) {
|
||||
for (let i of cust) {
|
||||
if (i.value == 0)
|
||||
continue;
|
||||
if (i.value < i.min || i.value > i.max) {
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error(`Expected range: ${i.min} ≤ ${i.name} ≤ ${i.max}, got ${i.value}`);
|
||||
}
|
||||
if (i.validator && !i.validator(i.value)) {
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error(`Invalid value: ${i.value}(${$i.name})\nValidator: ${i.validator}`);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function SaveCust(cust, buffer) {
|
||||
let dict = Object.assign({}, ...cust.map((x) => ({[x.name]: x})));
|
||||
let view = new DataView(buffer);
|
||||
for (let i of cust) {
|
||||
if (i.name == "custRev") {
|
||||
continue;
|
||||
}
|
||||
let id = i.name;
|
||||
i.value = document.getElementById(id).value;
|
||||
switch (i.size) {
|
||||
case 2:
|
||||
view.setUint16(i.offset, i.value, true);
|
||||
break;
|
||||
case 4:
|
||||
view.setUint32(i.offset, i.value, true);
|
||||
break;
|
||||
default:
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error("Unknown size at", i);
|
||||
}
|
||||
}
|
||||
ValidateCust(cust);
|
||||
let a = document.createElement("a");
|
||||
a.href = window.URL.createObjectURL(new Blob([buffer], {type: "application/octet-stream"}));
|
||||
a.download = "loader.kip";
|
||||
a.click();
|
||||
}
|
||||
|
||||
function UpdateHTMLForm(cust) {
|
||||
let dict = Object.assign({}, ...cust.map((x) => ({[x.name]: x})));
|
||||
let form = document.getElementById("form");
|
||||
for (let i of cust) {
|
||||
if (i.name == "custRev") {
|
||||
continue;
|
||||
}
|
||||
let id = i.name;
|
||||
let input = document.getElementById(id);
|
||||
if (!input) {
|
||||
let div = document.createElement("div");
|
||||
div.classList.add("form-floating", "my-3");
|
||||
let label = document.createElement("label");
|
||||
label.setAttribute("for", id);
|
||||
label.innerHTML = id;
|
||||
label.classList.add("form-entry");
|
||||
input = document.createElement("input");
|
||||
input.classList.add("form-control", "form-entry");
|
||||
input.min = dict[i.name].min;
|
||||
input.max = dict[i.name].max;
|
||||
input.id = id;
|
||||
input.type = "number";
|
||||
input.step = dict[i.name].step;
|
||||
div.appendChild(input);
|
||||
div.appendChild(label);
|
||||
|
||||
let desc = dict[i.name].desc;
|
||||
if (desc) {
|
||||
let tip = document.createElement("small");
|
||||
tip.innerHTML = desc;
|
||||
tip.classList.add("form-entry");
|
||||
tip.setAttribute("for", id);
|
||||
div.appendChild(tip);
|
||||
}
|
||||
|
||||
form.appendChild(div);
|
||||
}
|
||||
input.value = dict[i.name].value;
|
||||
}
|
||||
let btn = document.getElementById("save");
|
||||
if (!btn) {
|
||||
let div = document.createElement("div");
|
||||
div.classList.add("flex", "space-x-2", "justify-center");
|
||||
btn = document.createElement("button");
|
||||
btn.innerHTML = "Save Changes"
|
||||
btn.id = "save";
|
||||
btn.classList.add("save-btn");
|
||||
btn.addEventListener('click', () => {
|
||||
try {
|
||||
SaveCust(cust, buffer);
|
||||
} catch (e) {
|
||||
console.log(e);
|
||||
alert(e);
|
||||
}
|
||||
});
|
||||
div.appendChild(btn);
|
||||
form.appendChild(div);
|
||||
}
|
||||
}
|
||||
|
||||
function ParseCust(magicOffset, buffer) {
|
||||
let view = new DataView(buffer);
|
||||
let cust = InitCustTable();
|
||||
let offset = magicOffset + 4;
|
||||
for (let i of cust) {
|
||||
i.offset = offset;
|
||||
switch (i.size) {
|
||||
case 2:
|
||||
i.value = view.getUint16(offset, true);
|
||||
break;
|
||||
case 4:
|
||||
i.value = view.getUint32(offset, true);
|
||||
break;
|
||||
default:
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error("Unknown size at", i);
|
||||
}
|
||||
offset += i.size;
|
||||
}
|
||||
ValidateCust(cust);
|
||||
UpdateHTMLForm(cust);
|
||||
}
|
||||
|
||||
const fileInput = document.getElementById("file");
|
||||
fileInput.addEventListener('change', (event) => {
|
||||
let reader = new FileReader();
|
||||
reader.readAsArrayBuffer(event.target.files[0]);
|
||||
reader.onloadend = (progEvent) => {
|
||||
if (progEvent.target.readyState === FileReader.DONE) {
|
||||
buffer = progEvent.target.result;
|
||||
try {
|
||||
let offset = FindMagicOffset(buffer);
|
||||
ParseCust(offset, buffer);
|
||||
} catch (e) {
|
||||
console.log(e);
|
||||
alert(e);
|
||||
fileInput.value = "";
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
</script>
|
||||
</html>
|
||||
<script type="text/javascript" src="./main.js"></script>
|
||||
</html>
|
||||
|
||||
323
Source/loaderConfigurator/src/main.ts
Normal file
323
Source/loaderConfigurator/src/main.ts
Normal file
@@ -0,0 +1,323 @@
|
||||
const CUST_REV = 2;
|
||||
var buffer;
|
||||
|
||||
function FindMagicOffset(buffer) {
|
||||
let view = new DataView(buffer);
|
||||
for (let i = 0; i < view.byteLength; i += 4) {
|
||||
if (view.getUint32(i, true) == 0x54535543) { // "CUST"
|
||||
return i;
|
||||
}
|
||||
}
|
||||
throw new Error("Invalid loader.kip file");
|
||||
}
|
||||
|
||||
function CustEntry(name, size, desc, defval, minmax = [0, 1_000_000], step = 1, extra_validator = null) {
|
||||
this.name = name;
|
||||
this.size = size;
|
||||
this.desc = desc;
|
||||
this.defval = defval;
|
||||
this.min = minmax[0];
|
||||
this.max = minmax[1];
|
||||
this.step = step;
|
||||
this.validator = extra_validator;
|
||||
this.value = null;
|
||||
this.offset = null;
|
||||
}
|
||||
|
||||
function InitCustTable() {
|
||||
let cust = [
|
||||
new CustEntry(
|
||||
"mtcConf",
|
||||
2,
|
||||
"<b>DRAM Timing</b>\
|
||||
<li><b>0</b>: AUTO_ADJ_MARIKO_SAFE: Auto adjust timings for LPDDR4 ≤3733 Mbps specs, 8Gb density.</li>\
|
||||
<li><b>1</b>: AUTO_ADJ_MARIKO_4266: Auto adjust timings for LPDDR4X 4266 Mbps specs, 8Gb density.</li>\
|
||||
<li><b>2</b>: ENTIRE_TABLE_ERISTA: Not implemented.</li>\
|
||||
<li><b>3</b>: ENTIRE_TABLE_MARIKO: Not implemented.</li>",
|
||||
0,
|
||||
[0, 3],
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoCpuMaxClock",
|
||||
4,
|
||||
"<b>Mariko CPU Max Clock in kHz</b>\
|
||||
<li>System default: 1785000</li>\
|
||||
<li>≥ 2193000 will enable overvolting (> 1120 mV)</li>",
|
||||
2397_000,
|
||||
[1785_000, 3000_000],
|
||||
100,
|
||||
(x) => { return (x % 100) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoCpuBoostClock",
|
||||
4,
|
||||
"<b>Mariko CPU Boost Clock in kHz</b>\
|
||||
<li>System default: 1785000</li>\
|
||||
<li>Must not be higher than marikoCpuMaxClock</li>",
|
||||
1785_000,
|
||||
[1785_000, 3000_000],
|
||||
100,
|
||||
(x) => { return (x % 100) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoCpuMaxVolt",
|
||||
4,
|
||||
"<b>Mariko CPU Max Voltage in mV</b>\
|
||||
<li>System default: 1120</li>\
|
||||
<li>Acceptable range: 1100 ≤ x ≤ 1300</li>",
|
||||
1220,
|
||||
[1100, 1300],
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoGpuMaxClock",
|
||||
4,
|
||||
"<b>Mariko GPU Max Clock in kHz</b>\
|
||||
<li>System default: 921600</li>\
|
||||
<li>Tegra X1+ official maximum: 1267200</li>",
|
||||
1305_600,
|
||||
[768_000, 1536_000],
|
||||
100,
|
||||
(x) => { return (x % 100) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"marikoEmcMaxClock",
|
||||
4,
|
||||
"<b>Mariko RAM Max Clock in kHz</b>\
|
||||
<li>Values should be > 1600000, and divided evenly by 3200.</li>\
|
||||
<li><b>WARNING:</b> RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM</li>",
|
||||
1996_800,
|
||||
[1612_800, 2400_000],
|
||||
3200,
|
||||
(x) => { return (x % 3200) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"eristaCpuOCEnable",
|
||||
4,
|
||||
"<b>Erista CPU Enable Overclock</b>\
|
||||
<li>Not usable unless CPU cvb table is filled in</li>",
|
||||
0,
|
||||
[0, 1]
|
||||
),
|
||||
new CustEntry(
|
||||
"eristaCpuMaxVolt",
|
||||
4,
|
||||
"<b>Erista CPU Max Voltage in mV</b>\
|
||||
<li>Acceptable range: 1100 ≤ x ≤ 1400</li>\
|
||||
<li>Not enabled by default</li>",
|
||||
0,
|
||||
[0, 1400],
|
||||
100,
|
||||
(x) => { return x >= 1100; }
|
||||
),
|
||||
new CustEntry(
|
||||
"eristaEmcMaxClock",
|
||||
4,
|
||||
"<b>Erista RAM Max Clock in kHz</b>\
|
||||
<li>Values should be > 1600000, and divided evenly by 3200.</li>\
|
||||
<li><b>WARNING:</b> RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM</li>",
|
||||
1862_400,
|
||||
[1600_000, 2400_000],
|
||||
3200,
|
||||
(x) => { return (x % 3200) == 0; }
|
||||
),
|
||||
new CustEntry(
|
||||
"eristaEmcVolt",
|
||||
4,
|
||||
"<b>Erista RAM Voltage in uV</b>\
|
||||
<li>Acceptable range: 1100000 ≤ x ≤ 1250000, and it should be divided evenly by 12500.</li>\
|
||||
<li>Not enabled by default</li>",
|
||||
0,
|
||||
[0, 1250_000],
|
||||
12500,
|
||||
(x) => { return (x % 12500) == 0 && x >= 1100_000; }
|
||||
),
|
||||
];
|
||||
return cust;
|
||||
}
|
||||
|
||||
function ValidateCust(cust) {
|
||||
for (let i of cust) {
|
||||
if (i.value == 0)
|
||||
continue;
|
||||
if (i.value < i.min || i.value > i.max) {
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error(`Expected range: ${i.min} ≤ ${i.name} ≤ ${i.max}, got ${i.value}`);
|
||||
}
|
||||
if (i.validator && !i.validator(i.value)) {
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error(`Invalid value: ${i.value}(${i.name})\nValidator: ${i.validator}`);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function SaveCust(cust, buffer) {
|
||||
let dict = Object.assign({}, ...cust.map((x) => ({[x.name]: x})));
|
||||
let view = new DataView(buffer);
|
||||
let storage = {};
|
||||
for (let i of cust) {
|
||||
let id = i.name;
|
||||
i.value = (document.getElementById(id) as HTMLInputElement).value;
|
||||
storage[i.name] = i.value;
|
||||
switch (i.size) {
|
||||
case 2:
|
||||
view.setUint16(i.offset, i.value, true);
|
||||
break;
|
||||
case 4:
|
||||
view.setUint32(i.offset, i.value, true);
|
||||
break;
|
||||
default:
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error("Unknown size at " + i);
|
||||
}
|
||||
}
|
||||
ValidateCust(cust);
|
||||
storage["custRev"] = CUST_REV;
|
||||
localStorage.setItem("last_saved", JSON.stringify(storage));
|
||||
let a = document.createElement("a");
|
||||
a.href = window.URL.createObjectURL(new Blob([buffer], {type: "application/octet-stream"}));
|
||||
a.download = "loader.kip";
|
||||
a.click();
|
||||
}
|
||||
|
||||
function LastSaved() {
|
||||
let storage = localStorage.getItem("last_saved");
|
||||
if (!storage) {
|
||||
return false;
|
||||
}
|
||||
let sObj = JSON.parse(storage);
|
||||
if (!sObj["custRev"] || sObj["custRev"] != CUST_REV) {
|
||||
localStorage.removeItem("last_saved");
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
function LoadLastSaved() {
|
||||
if (LastSaved()) {
|
||||
let storage = localStorage.getItem("last_saved");
|
||||
let sObj = JSON.parse(storage);
|
||||
for (let key in sObj) {
|
||||
if (key == "custRev") {
|
||||
continue;
|
||||
}
|
||||
(document.getElementById(key) as HTMLInputElement).value = sObj[key];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function LoadDefault(cust) {
|
||||
let dict = Object.assign({}, ...cust.map((x) => ({[x.name]: x})));
|
||||
for (let i of cust) {
|
||||
let id = i.name;
|
||||
(document.getElementById(id) as HTMLInputElement).value = i.defval;
|
||||
}
|
||||
}
|
||||
|
||||
function UpdateHTMLForm(cust) {
|
||||
let dict = Object.assign({}, ...cust.map((x) => ({[x.name]: x})));
|
||||
let form = document.getElementById("form");
|
||||
for (let i of cust) {
|
||||
let id = i.name;
|
||||
let input = document.getElementById(id) as HTMLInputElement;
|
||||
if (!input) {
|
||||
let div = document.createElement("div");
|
||||
div.classList.add("form-floating", "my-3");
|
||||
let label = document.createElement("label");
|
||||
label.setAttribute("for", id);
|
||||
label.innerHTML = id;
|
||||
label.classList.add("form-entry");
|
||||
input = document.createElement("input");
|
||||
input.classList.add("form-control", "form-entry");
|
||||
input.min = dict[i.name].min;
|
||||
input.max = dict[i.name].max;
|
||||
input.id = id;
|
||||
input.type = "number";
|
||||
input.step = dict[i.name].step;
|
||||
div.appendChild(input);
|
||||
div.appendChild(label);
|
||||
|
||||
let desc = dict[i.name].desc;
|
||||
if (desc) {
|
||||
let tip = document.createElement("small");
|
||||
tip.innerHTML = desc;
|
||||
tip.classList.add("form-entry");
|
||||
tip.setAttribute("for", id);
|
||||
div.appendChild(tip);
|
||||
}
|
||||
|
||||
form.appendChild(div);
|
||||
}
|
||||
input.value = dict[i.name].value;
|
||||
}
|
||||
|
||||
let btn = document.getElementById("load");
|
||||
btn.classList.remove("hide");
|
||||
if (LastSaved()) {
|
||||
btn.innerHTML = "Load Last Saved";
|
||||
btn.addEventListener('click', () => {
|
||||
LoadLastSaved();
|
||||
});
|
||||
} else {
|
||||
btn.addEventListener('click', () => {
|
||||
LoadDefault(cust);
|
||||
});
|
||||
}
|
||||
|
||||
btn = document.getElementById("save");
|
||||
btn.classList.remove("hide");
|
||||
btn.addEventListener('click', () => {
|
||||
try {
|
||||
SaveCust(cust, buffer);
|
||||
} catch (e) {
|
||||
console.log(e);
|
||||
alert(e);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
function ParseCust(magicOffset, buffer) {
|
||||
let view = new DataView(buffer);
|
||||
let cust = InitCustTable();
|
||||
let offset = magicOffset + 4;
|
||||
let rev = view.getUint16(offset, true);
|
||||
if (rev != CUST_REV) {
|
||||
throw new Error("Unsupported custRev, expected: " + CUST_REV + ", got " + rev);
|
||||
}
|
||||
offset += 2;
|
||||
for (let i of cust) {
|
||||
i.offset = offset;
|
||||
switch (i.size) {
|
||||
case 2:
|
||||
i.value = view.getUint16(offset, true);
|
||||
break;
|
||||
case 4:
|
||||
i.value = view.getUint32(offset, true);
|
||||
break;
|
||||
default:
|
||||
document.getElementById(i.name).focus();
|
||||
throw new Error("Unknown size at " + i);
|
||||
}
|
||||
offset += i.size;
|
||||
}
|
||||
ValidateCust(cust);
|
||||
UpdateHTMLForm(cust);
|
||||
}
|
||||
|
||||
const fileInput = document.getElementById("file") as HTMLInputElement;
|
||||
fileInput.addEventListener('change', (event) => {
|
||||
let reader = new FileReader();
|
||||
reader.readAsArrayBuffer((event.target as HTMLInputElement).files[0]);
|
||||
reader.onloadend = (progEvent) => {
|
||||
if (progEvent.target.readyState === FileReader.DONE) {
|
||||
buffer = progEvent.target.result;
|
||||
try {
|
||||
let offset = FindMagicOffset(buffer);
|
||||
ParseCust(offset, buffer);
|
||||
} catch (e) {
|
||||
console.log(e);
|
||||
alert(e);
|
||||
fileInput.value = "";
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
@@ -3,17 +3,37 @@
|
||||
@tailwind utilities;
|
||||
|
||||
label.form-entry {
|
||||
@apply inline-block mb-2 text-gray-700;
|
||||
@apply inline-block mb-2 text-slate-700 dark:text-slate-200;
|
||||
}
|
||||
|
||||
input.form-entry {
|
||||
@apply block w-full px-3 py-1.5 text-base font-normal text-gray-700 bg-white bg-clip-padding border border-solid border-gray-300 rounded transition ease-in-out m-0 focus:text-gray-700 focus:bg-white focus:border-blue-600 focus:outline-none;
|
||||
@apply block w-full px-3 py-1.5 text-base font-normal text-slate-600 dark:text-slate-300 bg-slate-100 dark:bg-slate-900 bg-clip-padding border border-solid border-slate-300 dark:border-gray-600 rounded transition ease-in-out m-0 focus:text-slate-700 dark:focus:text-slate-200 focus:bg-slate-100 dark:focus:bg-slate-800 focus:border-blue-600 dark:focus:border-blue-300 focus:outline-none;
|
||||
}
|
||||
|
||||
small.form-entry {
|
||||
@apply block mt-1 text-xs text-gray-600 px-3 py-1.5;
|
||||
@apply block mt-1 text-xs text-slate-600 dark:text-slate-400 px-3 py-1.5;
|
||||
}
|
||||
|
||||
button.save-btn {
|
||||
@apply inline-block px-6 py-2.5 bg-blue-600 text-white font-medium text-xs leading-tight uppercase rounded shadow-md hover:bg-blue-700 hover:shadow-lg focus:bg-blue-700 focus:shadow-lg focus:outline-none focus:ring-0 active:bg-blue-800 active:shadow-lg transition duration-150 ease-in-out;
|
||||
}
|
||||
a.link-primary {
|
||||
@apply text-blue-500 hover:text-blue-600 active:text-blue-700 transition duration-300 ease-in-out mb-4;
|
||||
}
|
||||
|
||||
div.btn {
|
||||
@apply grid grid-cols-2 gap-x-6 mt-3;
|
||||
}
|
||||
|
||||
button.btn {
|
||||
@apply inline-block px-6 py-2.5 text-slate-100 font-medium text-xs leading-tight uppercase rounded shadow-md hover:shadow-lg focus:shadow-lg focus:outline-none focus:ring-0 active:shadow-lg transition duration-150 ease-in-out;
|
||||
}
|
||||
|
||||
button.btn-primary {
|
||||
@apply bg-blue-600 hover:bg-blue-700 focus:bg-blue-700 active:bg-blue-900;
|
||||
}
|
||||
|
||||
button.btn-secondary {
|
||||
@apply bg-slate-600 hover:bg-slate-700 focus:bg-slate-700 active:bg-slate-900;
|
||||
}
|
||||
|
||||
.hide {
|
||||
display: none !important;
|
||||
}
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
/** @type {import('tailwindcss').Config} */
|
||||
module.exports = {
|
||||
content: ['./src/*.html', './node_modules/tw-elements/dist/js/**/*.js'],
|
||||
content: ['./dist/*.html', './dist/*.js', './node_modules/tw-elements/dist/js/**/*.js'],
|
||||
theme: {
|
||||
extend: {},
|
||||
},
|
||||
|
||||
Reference in New Issue
Block a user