var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; /* Config: Cust */ const CUST_REV = 3; var buffer; var CustPlatform; (function (CustPlatform) { CustPlatform[CustPlatform["Undefined"] = 0] = "Undefined"; CustPlatform[CustPlatform["Erista"] = 1] = "Erista"; CustPlatform[CustPlatform["Mariko"] = 2] = "Mariko"; CustPlatform[CustPlatform["All"] = 3] = "All"; })(CustPlatform || (CustPlatform = {})); ; class CustEntry { constructor(id, name, platform, size, desc, defval, minmax = [0, 1000000], step = 1, zeroable = true) { this.id = id; this.name = name; this.platform = platform; this.size = size; this.desc = desc; this.defval = defval; this.min = minmax[0]; this.max = minmax[1]; this.step = step; this.zeroable = zeroable; } ; validate() { let tip = new ErrorToolTip(this.id); tip.clear(); if (Number.isNaN(this.value) || !this.value) { tip.setMsg(`Invalid value: Not a number`); tip.show(); return false; } if (this.zeroable && this.value == 0) return true; if (this.value < this.min || this.value > this.max) { tip.setMsg(`Expected range: [${this.min}, ${this.max}], got ${this.value}.`); tip.show(); return false; } if (this.value % this.step != 0) { tip.setMsg(`${this.value} % ${this.step} ≠ 0`); tip.show(); return false; } return true; } ; getInputElement() { return document.getElementById(this.id); } updateValueFromElement() { this.value = Number(this.getInputElement().value); } isAvailableFor(platform) { return platform === CustPlatform.Undefined || this.platform === platform || this.platform === CustPlatform.All; } createElement() { let form = document.getElementById("form"); let input = this.getInputElement(); if (!input) { let grid = document.createElement("div"); grid.classList.add("grid", "cust-element"); // Label and input input = document.createElement("input"); input.min = String(this.zeroable ? 0 : this.min); input.max = String(this.max); input.id = this.id; input.type = "number"; input.step = String(this.step); let label = document.createElement("label"); label.setAttribute("for", this.id); label.innerHTML = this.name; label.appendChild(input); grid.appendChild(label); // Description in blockquote style let desc = document.createElement("blockquote"); desc.innerHTML = this.desc; desc.setAttribute("for", this.id); grid.appendChild(desc); form.appendChild(grid); let tooltip = new ErrorToolTip(this.id); tooltip.addChangeListener(); } input.value = String(this.value); } setElementDefaultValue() { document.getElementById(this.id).value = String(this.defval); } removeElement() { let input = this.getInputElement(); if (input) { input.parentElement.parentElement.remove(); } } showElement() { let input = this.getInputElement(); if (input) { input.parentElement.parentElement.style.removeProperty("display"); } } hideElement() { let input = this.getInputElement(); if (input) { input.parentElement.parentElement.style.setProperty("display", "none"); } } } var CustTable = [ new CustEntry("mtcConf", "DRAM Timing", CustPlatform.Mariko, 2, "
  • 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: NO_ADJ_ALL: No timing adjustment for both Erista and Mariko. Might achieve better performance on Mariko but lower maximum frequency is expected.", 0, [0, 2], 1), new CustEntry("marikoCpuMaxClock", "Mariko CPU Max Clock in kHz", CustPlatform.Mariko, 4, "
  • System default: 1785000
  • \
  • 2397000 might be unreachable for some SoCs.
  • ", 2397000, [1785000, 3000000], 1), new CustEntry("marikoCpuBoostClock", "Mariko CPU Boost Clock in kHz", CustPlatform.Mariko, 4, "
  • System default: 1785000
  • \
  • Boost clock will be applied when applications request higher CPU frequency for quicker loading.
  • \
  • This will be set regardless of whether sys-clk is enabled.
  • ", 1785000, [1020000, 3000000], 1, false), new CustEntry("marikoCpuMaxVolt", "Mariko CPU Max Voltage in mV", CustPlatform.Mariko, 4, "
  • System default: 1120
  • \
  • Acceptable range: 1100 ≤ x ≤ 1300
  • ", 1235, [1100, 1300], 5), new CustEntry("marikoGpuMaxClock", "Mariko GPU Max Clock in kHz", CustPlatform.Mariko, 4, "
  • System default: 921600
  • \
  • Tegra X1+ official maximum: 1267200
  • \
  • 1305600 might be unreachable for some SoCs.
  • ", 1305600, [768000, 1536000], 100), new CustEntry("marikoEmcMaxClock", "Mariko RAM Max Clock in kHz", CustPlatform.Mariko, 4, "
  • Values should be ≥ 1600000, and divided evenly by 3200.
  • \
  • WARNING: RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM
  • ", 1996800, [1600000, 2400000], 3200), new CustEntry("marikoEmcVddqVolt", "EMC Vddq (Mariko Only) Voltage in uV", CustPlatform.Mariko, 4, "
  • Acceptable range: 550000 ≤ x ≤ 650000
  • \
  • Value should be divided evenly by 5000
  • \
  • Default: 600000
  • \
  • Not enabled by default.
  • \
  • This will not work without sys-clk-OC.
  • ", 0, [550000, 650000], 5000), new CustEntry("eristaCpuMaxVolt", "Erista CPU Max Voltage in mV", CustPlatform.Erista, 4, "
  • Acceptable range: 1100 ≤ x ≤ 1300
  • ", 1235, [1100, 1300], 1), new CustEntry("eristaEmcMaxClock", "Erista RAM Max Clock in kHz", CustPlatform.Erista, 4, "
  • Values should be ≥ 1600000, and divided evenly by 3200.
  • \
  • WARNING: RAM overclock could be UNSTABLE if timing parameters are not suitable for your DRAM
  • ", 1862400, [1600000, 2400000], 3200), new CustEntry("commonEmcMemVolt", "EMC Vddq (Erista Only) & RAM Vdd2 Voltage in uV", CustPlatform.All, 4, "
  • Acceptable range: 1100000 ≤ x ≤ 1250000, and it should be divided evenly by 12500.
  • \
  • Erista Default (HOS): 1125000 (bootloader: 1100000)
  • \
  • Mariko Default: 1100000 (It will not work without sys-clk-OC)
  • \
  • Not enabled by default
  • ", 0, [1100000, 1250000], 12500), ]; 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"); } class ErrorToolTip { constructor(id, msg) { this.id = id; this.element = document.getElementById(id); if (msg) { this.setMsg(msg); } } ; setMsg(msg) { this.message = msg; } show() { if (this.element) { this.element.setAttribute("aria-invalid", "true"); if (this.message) { this.element.setAttribute("title", this.message); this.element.parentElement.setAttribute("data-tooltip", this.message); this.element.parentElement.setAttribute("data-placement", "top"); } } } ; clear() { if (this.element) { this.element.removeAttribute("aria-invalid"); this.element.removeAttribute("title"); this.element.parentElement.removeAttribute("data-tooltip"); this.element.parentElement.removeAttribute("data-placement"); } } addChangeListener() { if (this.element) { this.element.addEventListener('change', (_evt) => { let obj = CustTable.filter((obj) => { return obj.id === this.id; })[0]; obj.value = Number(this.element.value); obj.validate(); }); } } } ; function SaveCust(buffer) { let view = new DataView(buffer); let storage = {}; CustTable.forEach(i => { i.updateValueFromElement(); if (!i.validate() || !i.value) { document.getElementById(i.id).focus(); throw new Error(`Invalid ${i.name}`); } if (!i.offset) { document.getElementById(i.id).focus(); throw new Error(`Failed to get offset for ${i.name}`); ; } 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.id).focus(); throw new Error(`Unknown size at ${i.name}`); } storage[i.id] = i.value; }); 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 CustNavTabsInit() { const custNavTabs = Array.from(document.querySelectorAll(`nav[role="tab-control"] > button`)); custNavTabs.forEach(i => { i.removeAttribute("disabled"); let platform = Number(i.getAttribute("data-filter")); i.addEventListener('click', (_evt) => { const unfocusedClasses = ["outline", "secondary"]; i.classList.remove(...unfocusedClasses); custNavTabs.filter(j => j != i).forEach(k => k.classList.add(...unfocusedClasses)); CustTable.forEach(e => { if (e.isAvailableFor(platform)) { e.showElement(); } else { e.hideElement(); } }); }); }); } function UpdateHTMLForm() { CustTable.forEach(i => i.createElement()); let default_btn = document.getElementById("load_default"); default_btn.removeAttribute("disabled"); default_btn.addEventListener('click', () => { CustTable.forEach(i => i.setElementDefaultValue()); }); let last_btn = document.getElementById("load_saved"); if (LastSaved()) { last_btn.style.removeProperty("display"); last_btn.removeAttribute("disabled"); last_btn.addEventListener('click', () => { // Load last saved from localStorage JSON.parse(localStorage.getItem("last_saved")) .filter((key) => key != "custRev") .forEach((key) => document.getElementById(key).value = key); }); } else { last_btn.style.setProperty("display", "none"); } let save_btn = document.getElementById("save"); save_btn.removeAttribute("disabled"); save_btn.addEventListener('click', () => { try { SaveCust(buffer); } catch (e) { console.log(e); alert(e); } }); } function ParseCust(magicOffset, buffer) { let view = new DataView(buffer); let offset = magicOffset + 4; let rev = view.getUint16(offset, true); offset += 2; if (rev != CUST_REV) { throw new Error(`Unsupported custRev, expected: ${CUST_REV}, got ${rev}`); } document.getElementById("cust_rev").innerHTML = `Cust V${CUST_REV} is loaded.`; CustTable.forEach(i => { 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.id).focus(); throw new Error("Unknown size at " + i); } offset += i.size; i.validate(); }); } 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); CustTable.forEach(i => i.removeElement()); ParseCust(offset, buffer); CustNavTabsInit(); UpdateHTMLForm(); } catch (e) { console.log(e); alert(e); fileInput.value = ""; } } }; }); function fetchRelease() { return __awaiter(this, void 0, void 0, function* () { try { const responseFromSuite = yield fetch('https://api.github.com/repos/KazushiMe/Switch-OC-Suite/releases/latest', { method: 'GET', headers: { Accept: 'application/json', }, }); if (!responseFromSuite.ok) { throw new Error(`Failed to fetch latest release info from GitHub: ${responseFromSuite.status}`); } const resultFromSuite = yield responseFromSuite.json(); const latestVerFromSuite = resultFromSuite.tag_name; const amsVer = latestVerFromSuite.split(".").slice(0, 3).join("."); const loaderKip = resultFromSuite.assets.filter((obj) => { return obj.name.endsWith("loader.kip"); })[0]; const sdOut = resultFromSuite.assets.filter((obj) => { return obj.name.endsWith(".zip"); })[0]; const amsReleaseUrl = `https://github.com/Atmosphere-NX/Atmosphere/releases/tags/${amsVer}`; let info = { OCSuiteVer: latestVerFromSuite, LoaderKipUrl: loaderKip.browser_download_url, LoaderKipTime: loaderKip.updated_at, SdOutZipUrl: sdOut.browser_download_url, SdOutZipTime: sdOut.updated_at, AMSVer: amsVer, AMSUrl: amsReleaseUrl }; return info; } catch (e) { console.log(e); alert(e); } }); } function updateDownloadUrls() { return __awaiter(this, void 0, void 0, function* () { const updateHref = (id, name, url) => { let element = document.getElementById(id); element.innerHTML = name; element.removeAttribute("aria-busy"); element.setAttribute("href", url); }; let info = yield fetchRelease(); if (info) { const loaderKipName = `loader.kip ${info.OCSuiteVer}
    ${info.LoaderKipTime}`; updateHref("loader_kip_btn", loaderKipName, info.LoaderKipUrl); const sdOutName = `SdOut.zip ${info.OCSuiteVer}
    ${info.SdOutZipTime}`; updateHref("sdout_zip_btn", sdOutName, info.SdOutZipUrl); const amsName = `Atmosphere-NX ${info.AMSVer}`; updateHref("ams_btn", amsName, info.AMSUrl); } }); } addEventListener('DOMContentLoaded', (_evt) => __awaiter(this, void 0, void 0, function* () { yield updateDownloadUrls(); }));