Files
driver-manager/webroot/index.html

401 lines
16 KiB
HTML
Raw Normal View History

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Driver Manager</title>
<style>
:root {
--bg: #0d1117;
--surface: #161b22;
--surface2: #21262d;
--border: #30363d;
--primary: #58a6ff;
--primary-dim: #388bfd;
--green: #3fb950;
--orange: #d29922;
--red: #f85149;
--text: #c9d1d9;
--text-dim: #8b949e;
--radius: 10px;
}
* { margin: 0; padding: 0; box-sizing: border-box; }
body {
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
background: var(--bg);
color: var(--text);
padding: 16px;
-webkit-font-smoothing: antialiased;
}
.header { text-align: center; padding: 20px 0 12px; }
.header h1 { font-size: 20px; color: var(--primary); }
.header .sub { font-size: 12px; color: var(--text-dim); margin-top: 4px; }
.card {
background: var(--surface);
border: 1px solid var(--border);
border-radius: var(--radius);
padding: 14px;
margin-bottom: 10px;
}
.card-title {
font-size: 12px;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.8px;
color: var(--primary);
margin-bottom: 10px;
display: flex;
align-items: center;
gap: 6px;
}
.card-title .dot {
width: 8px; height: 8px;
border-radius: 50%;
background: var(--green);
}
.card-title .dot.off { background: var(--red); }
.card-title .dot.warn { background: var(--orange); }
.row {
display: flex;
align-items: center;
justify-content: space-between;
padding: 10px 0;
border-bottom: 1px solid var(--border);
}
.row:last-child { border-bottom: none; }
.row-label { font-size: 14px; font-weight: 500; }
.row-desc { font-size: 11px; color: var(--text-dim); margin-top: 2px; }
.row-value { font-size: 13px; color: var(--green); font-family: monospace; }
select.sel {
background: var(--surface2);
color: var(--text);
border: 1px solid var(--border);
border-radius: 6px;
padding: 6px 10px;
font-size: 13px;
outline: none;
}
.info-grid {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 6px;
}
.info-item {
background: var(--surface2);
border-radius: 6px;
padding: 8px 10px;
}
.info-item .lbl { font-size: 10px; color: var(--text-dim); text-transform: uppercase; letter-spacing: 0.3px; }
.info-item .val { font-size: 13px; font-weight: 500; margin-top: 2px; font-family: monospace; }
.log-box {
background: var(--surface2);
border-radius: 6px;
padding: 10px;
font-family: "Cascadia Code", "Fira Code", monospace;
font-size: 11px;
line-height: 1.5;
max-height: 180px;
overflow-y: auto;
color: var(--text-dim);
white-space: pre-wrap;
word-break: break-all;
}
.btn-row { display: flex; gap: 6px; margin-top: 8px; }
.btn {
flex: 1;
padding: 8px;
border: 1px solid var(--border);
border-radius: 6px;
background: var(--surface2);
color: var(--text);
font-size: 12px;
font-weight: 500;
cursor: pointer;
text-align: center;
}
.btn:active { opacity: 0.7; }
.btn-primary { background: var(--primary-dim); color: #fff; border-color: var(--primary); }
</style>
</head>
<body>
<div class="header">
<h1>Driver Manager</h1>
<div class="sub">GPU / WiFi / Bluetooth / SDR / Controllers</div>
</div>
<!-- Hardware Info -->
<div class="card">
<div class="card-title"><span class="dot"></span> Hardware</div>
<div class="info-grid">
<div class="info-item"><div class="lbl">Device</div><div class="val" id="hwDevice"></div></div>
<div class="info-item"><div class="lbl">SoC</div><div class="val" id="hwSoc"></div></div>
<div class="info-item"><div class="lbl">GPU</div><div class="val" id="hwGpu"></div></div>
<div class="info-item"><div class="lbl">WiFi</div><div class="val" id="hwWifi"></div></div>
<div class="info-item"><div class="lbl">Bluetooth</div><div class="val" id="hwBt"></div></div>
<div class="info-item"><div class="lbl">USB Devices</div><div class="val" id="hwUsb"></div></div>
</div>
</div>
<!-- GPU -->
<div class="card">
<div class="card-title"><span class="dot" id="gpuDot"></span> GPU — PowerVR</div>
<div class="row">
<div>
<div class="row-label">Driver Mode</div>
<div class="row-desc">Vulkan 1.4 / OpenGL ES 3.x / OpenCL 3.0</div>
</div>
<select class="sel" id="gpuMode" onchange="setMode('gpu_mode', this.value)">
<option value="performance">Performance</option>
<option value="balanced">Balanced</option>
<option value="powersave">Power Save</option>
<option value="compute">Compute (OpenCL)</option>
</select>
</div>
<div class="row">
<div><div class="row-label">Vulkan Version</div></div>
<div class="row-value" id="gpuVulkan"></div>
</div>
<div class="row">
<div><div class="row-label">GL Renderer</div></div>
<div class="row-value" id="gpuRenderer"></div>
</div>
</div>
<!-- WiFi -->
<div class="card">
<div class="card-title"><span class="dot" id="wifiDot"></span> WiFi — BCM4390</div>
<div class="row">
<div>
<div class="row-label">Driver Mode</div>
<div class="row-desc">Nexmon: monitor + injection support</div>
</div>
<select class="sel" id="wifiMode" onchange="setMode('wifi_mode', this.value)">
<option value="standard">Standard</option>
<option value="monitor">Monitor (Nexmon)</option>
<option value="injection">Injection (Nexmon)</option>
<option value="restore">Restore Stock</option>
</select>
</div>
<div class="row">
<div><div class="row-label">Interface</div></div>
<div class="row-value" id="wifiIface"></div>
</div>
<div class="row">
<div><div class="row-label">Nexmon Firmware</div></div>
<div class="row-value" id="wifiNexmon"></div>
</div>
</div>
<!-- Bluetooth -->
<div class="card">
<div class="card-title"><span class="dot" id="btDot"></span> Bluetooth — QCA</div>
<div class="row">
<div>
<div class="row-label">Driver Mode</div>
<div class="row-desc">Standard / pentest (raw HCI, all profiles)</div>
</div>
<select class="sel" id="btMode" onchange="setMode('bt_mode', this.value)">
<option value="standard">Standard</option>
<option value="pentest">Pentest</option>
<option value="disabled">Disabled</option>
</select>
</div>
</div>
<!-- SDR -->
<div class="card">
<div class="card-title"><span class="dot" id="sdrDot"></span> SDR Drivers</div>
<div class="row">
<div>
<div class="row-label">Radio Mode</div>
<div class="row-desc">RTL-SDR v1-4 / HackRF / Airspy / LimeSDR</div>
</div>
<select class="sel" id="sdrMode" onchange="setMode('sdr_mode', this.value)">
<option value="sdr">SDR Scanner</option>
<option value="dvbt">DVB-T (Digital TV)</option>
<option value="hackrf">HackRF (TX/RX)</option>
<option value="off">Off</option>
</select>
</div>
<div class="row">
<div>
<div class="row-label">Decoder</div>
<div class="row-desc">Background signal decoder</div>
</div>
<select class="sel" id="decoderMode" onchange="setMode('decoder_mode', this.value)">
<option value="off">Off</option>
<option value="adsb">ADS-B (1090 MHz)</option>
<option value="fm">FM Radio</option>
<option value="spectrum">Spectrum Scan</option>
</select>
</div>
<div class="row">
<div><div class="row-label">USB SDR Devices</div></div>
<div class="row-value" id="sdrDevices"></div>
</div>
</div>
<!-- Game Controllers -->
<div class="card">
<div class="card-title"><span class="dot" id="padDot"></span> Game Controllers</div>
<div class="row">
<div>
<div class="row-label">Controller Mode</div>
<div class="row-desc">All native: Xbox, PS5, PS4, Switch, Steam, Logitech, 8BitDo, Wacom</div>
</div>
<select class="sel" id="gamepadMode" onchange="setMode('gamepad_mode', this.value)">
<option value="auto">Auto (All)</option>
<option value="xbox">Xbox Only</option>
<option value="playstation">PlayStation Only</option>
<option value="nintendo">Nintendo Only</option>
<option value="off">Off</option>
</select>
</div>
<div class="row">
<div><div class="row-label">Connected Controllers</div></div>
<div class="row-value" id="padDevices"></div>
</div>
</div>
<!-- Log -->
<div class="card">
<div class="card-title">Log</div>
<div class="log-box" id="logArea">Initializing...</div>
<div class="btn-row">
<button class="btn" onclick="refreshAll()">Refresh</button>
<button class="btn" onclick="showLog()">Module Log</button>
<button class="btn btn-primary" onclick="applyAll()">Apply & Restart</button>
</div>
</div>
<script>
const MODDIR = '/data/adb/modules/driver-manager';
function exec(cmd) {
return new Promise((resolve, reject) => {
const cb = 'cb_' + Date.now() + '_' + Math.random().toString(36).slice(2);
window[cb] = (errno, stdout, stderr) => {
delete window[cb];
resolve({ errno, stdout: stdout || '', stderr: stderr || '' });
};
try { ksu.exec(cmd, '{}', cb); }
catch (e) { delete window[cb]; reject(e); }
});
}
function log(msg) {
const a = document.getElementById('logArea');
const ts = new Date().toLocaleTimeString('en-US', { hour12: false });
a.textContent += '\n[' + ts + '] ' + msg;
a.scrollTop = a.scrollHeight;
}
async function gp(name) {
try { return (await exec('getprop ' + name)).stdout.trim(); }
catch (e) { return ''; }
}
async function setMode(file, value) {
log('Setting ' + file + ' = ' + value);
await exec('echo "' + value + '" > ' + MODDIR + '/config/' + file);
log('OK. Changes apply on next boot or click Apply & Restart.');
}
async function applyAll() {
log('Restarting driver service...');
await exec('sh ' + MODDIR + '/service.sh &');
log('Service restarted. Some changes need reboot.');
setTimeout(refreshAll, 2000);
}
async function loadHardware() {
document.getElementById('hwDevice').textContent = await gp('ro.product.model');
document.getElementById('hwSoc').textContent = await gp('ro.soc.model');
document.getElementById('hwGpu').textContent = await gp('ro.hardware.egl');
const wifi = (await exec('ls /sys/module/ | grep -iE "^(bcmdhd|qca|wcn|ath|mt76)" | head -1')).stdout.trim();
document.getElementById('hwWifi').textContent = wifi || 'unknown';
const bt = (await exec('ls /sys/module/ | grep -iE "^(btqca|btusb|btbcm)" | head -1')).stdout.trim();
document.getElementById('hwBt').textContent = bt || 'unknown';
const usb = (await exec('lsusb 2>/dev/null | wc -l || echo 0')).stdout.trim();
document.getElementById('hwUsb').textContent = usb + ' devices';
}
async function loadModes() {
const modes = ['gpu_mode', 'wifi_mode', 'bt_mode', 'sdr_mode', 'gamepad_mode', 'decoder_mode'];
const ids = ['gpuMode', 'wifiMode', 'btMode', 'sdrMode', 'gamepadMode', 'decoderMode'];
for (let i = 0; i < modes.length; i++) {
const val = (await exec('cat ' + MODDIR + '/config/' + modes[i] + ' 2>/dev/null')).stdout.trim();
if (val) document.getElementById(ids[i]).value = val;
}
}
async function loadGpuInfo() {
const renderer = await gp('debug.hwui.renderer');
document.getElementById('gpuRenderer').textContent = renderer || 'default';
const vulkan = await gp('ro.hardware.vulkan');
document.getElementById('gpuVulkan').textContent = vulkan ? vulkan + ' (1.4)' : 'unknown';
}
async function loadWifiInfo() {
const iface = (await exec('ip link show wlan0 2>/dev/null | head -1')).stdout.trim();
document.getElementById('wifiIface').textContent = iface ? 'wlan0 (up)' : 'wlan0';
const nexmon = (await exec('ls ' + MODDIR + '/firmware/fw_bcm4390_*.bin 2>/dev/null | wc -l')).stdout.trim();
document.getElementById('wifiNexmon').textContent = nexmon > 0 ? nexmon + ' firmware(s)' : 'not installed';
}
async function loadSdrInfo() {
// Detect connected USB SDR devices by vendor:product ID
const usb = (await exec(
'cat /sys/bus/usb/devices/*/idVendor 2>/dev/null | while read v; do ' +
'dir=$(grep -rl "$v" /sys/bus/usb/devices/*/idVendor 2>/dev/null | head -1 | xargs dirname); ' +
'p=$(cat "$dir/idProduct" 2>/dev/null); ' +
'case "$v:$p" in ' +
'"0bda:2832"|"0bda:2838") echo "RTL-SDR";; ' +
'"1d50:6089") echo "HackRF";; ' +
'"1d50:60a1") echo "Airspy";; ' +
'"0403:6014"|"04b4:00f3") echo "LimeSDR";; ' +
'esac; done 2>/dev/null | sort -u | tr "\\n" ", "'
)).stdout.trim();
document.getElementById('sdrDevices').textContent = usb || 'none detected';
}
async function loadControllerInfo() {
const pads = (await exec('cat /proc/bus/input/devices 2>/dev/null | grep -iE "(xbox|playstation|dualsense|pro controller|8bitdo|gamepad)" | wc -l')).stdout.trim();
document.getElementById('padDevices').textContent = (pads && pads !== '0') ? pads + ' connected' : 'none';
}
async function showLog() {
const r = await exec('cat ' + MODDIR + '/driver-manager.log 2>/dev/null || echo "No log"');
document.getElementById('logArea').textContent = '=== Module Log ===\n' + r.stdout;
}
async function refreshAll() {
log('Refreshing...');
await loadHardware();
await loadModes();
await loadGpuInfo();
await loadWifiInfo();
await loadSdrInfo();
await loadControllerInfo();
log('Done');
}
(async () => {
try { await refreshAll(); log('Ready'); }
catch (e) { log('Init error: ' + e.message); }
})();
</script>
</body>
</html>