From c1668a4e4aa31b5693542a6f254856b158d71a48 Mon Sep 17 00:00:00 2001 From: Louis Date: Thu, 7 Aug 2025 19:59:09 +0700 Subject: [PATCH] refactor: clean up unused hardware apis --- .../template-tauri-build-linux-x64.yml | 3 +- Makefile | 3 - package.json | 1 - scripts/download-lib.mjs | 86 ------- src-tauri/icons/icon.png | Bin 38562 -> 36896 bytes .../src/core/{hardware/mod.rs => hardware.rs} | 148 +----------- src-tauri/src/core/hardware/amd.rs | 210 ------------------ src-tauri/src/core/hardware/nvidia.rs | 120 ---------- src-tauri/src/core/hardware/vulkan.rs | 145 ------------ src-tauri/tauri.bundle.windows.nsis.template | 2 - src-tauri/tauri.linux.conf.json | 3 +- testRunner.js | 19 -- web-app/src/hooks/useHardware.ts | 70 ------ web-app/src/routes/hub/index.tsx | 2 +- web-app/src/routes/system-monitor.tsx | 17 +- 15 files changed, 10 insertions(+), 819 deletions(-) delete mode 100644 scripts/download-lib.mjs rename src-tauri/src/core/{hardware/mod.rs => hardware.rs} (60%) delete mode 100644 src-tauri/src/core/hardware/amd.rs delete mode 100644 src-tauri/src/core/hardware/nvidia.rs delete mode 100644 src-tauri/src/core/hardware/vulkan.rs delete mode 100644 testRunner.js diff --git a/.github/workflows/template-tauri-build-linux-x64.yml b/.github/workflows/template-tauri-build-linux-x64.yml index 20663ea69..d7eeaefec 100644 --- a/.github/workflows/template-tauri-build-linux-x64.yml +++ b/.github/workflows/template-tauri-build-linux-x64.yml @@ -105,8 +105,7 @@ jobs: jq --arg version "${{ inputs.new_version }}" '.version = $version | .bundle.createUpdaterArtifacts = true' ./src-tauri/tauri.conf.json > /tmp/tauri.conf.json mv /tmp/tauri.conf.json ./src-tauri/tauri.conf.json if [ "${{ inputs.channel }}" != "stable" ]; then - jq '.bundle.linux.deb.files = {"usr/bin/bun": "resources/bin/bun", - "usr/lib/Jan-${{ inputs.channel }}/resources/lib/libvulkan.so": "resources/lib/libvulkan.so"}' ./src-tauri/tauri.linux.conf.json > /tmp/tauri.linux.conf.json + jq '.bundle.linux.deb.files = {"usr/bin/bun": "resources/bin/bun"}' ./src-tauri/tauri.linux.conf.json > /tmp/tauri.linux.conf.json mv /tmp/tauri.linux.conf.json ./src-tauri/tauri.linux.conf.json fi jq --arg version "${{ inputs.new_version }}" '.version = $version' web-app/package.json > /tmp/package.json diff --git a/Makefile b/Makefile index 023f2c877..8ffbff54a 100644 --- a/Makefile +++ b/Makefile @@ -31,7 +31,6 @@ endif dev: install-and-build yarn download:bin - yarn download:lib yarn dev # Linting @@ -41,7 +40,6 @@ lint: install-and-build # Testing test: lint yarn download:bin - yarn download:lib yarn test # Builds and publishes the app @@ -50,7 +48,6 @@ build-and-publish: install-and-build # Build build: install-and-build - yarn download:lib yarn build clean: diff --git a/package.json b/package.json index 99bf81631..fab394131 100644 --- a/package.json +++ b/package.json @@ -19,7 +19,6 @@ "dev:web": "yarn workspace @janhq/web-app dev", "dev:tauri": "yarn build:icon && yarn copy:assets:tauri && cross-env IS_CLEAN=true tauri dev", "copy:assets:tauri": "cpx \"pre-install/*.tgz\" \"src-tauri/resources/pre-install/\"", - "download:lib": "node ./scripts/download-lib.mjs", "download:bin": "node ./scripts/download-bin.mjs", "build:tauri:win32": "yarn download:bin && yarn tauri build", "build:tauri:linux": "yarn download:bin && ./src-tauri/build-utils/shim-linuxdeploy.sh yarn tauri build && ./src-tauri/build-utils/buildAppImage.sh", diff --git a/scripts/download-lib.mjs b/scripts/download-lib.mjs deleted file mode 100644 index d2086b36e..000000000 --- a/scripts/download-lib.mjs +++ /dev/null @@ -1,86 +0,0 @@ -console.log('Script is running') -// scripts/download-lib.mjs -import https from 'https' -import fs, { mkdirSync } from 'fs' -import os from 'os' -import path from 'path' -import { copySync } from 'cpx' - -function download(url, dest) { - return new Promise((resolve, reject) => { - console.log(`Downloading ${url} to ${dest}`) - const file = fs.createWriteStream(dest) - https - .get(url, (response) => { - console.log(`Response status code: ${response.statusCode}`) - if ( - response.statusCode >= 300 && - response.statusCode < 400 && - response.headers.location - ) { - // Handle redirect - const redirectURL = response.headers.location - console.log(`Redirecting to ${redirectURL}`) - download(redirectURL, dest).then(resolve, reject) // Recursive call - return - } else if (response.statusCode !== 200) { - reject(`Failed to get '${url}' (${response.statusCode})`) - return - } - response.pipe(file) - file.on('finish', () => { - file.close(resolve) - }) - }) - .on('error', (err) => { - fs.unlink(dest, () => reject(err.message)) - }) - }) -} - -async function main() { - console.log('Starting main function') - const platform = os.platform() // 'darwin', 'linux', 'win32' - const arch = os.arch() // 'x64', 'arm64', etc. - - if (arch != 'x64') return - - let filename - if (platform == 'linux') - filename = 'libvulkan.so' - else if (platform == 'win32') - filename = 'vulkan-1.dll' - else - return - - const url = `https://catalog.jan.ai/${filename}` - - const libDir = 'src-tauri/resources/lib' - const tempDir = 'scripts/dist' - - try { - mkdirSync('scripts/dist') - } catch (err) { - // Expect EEXIST error if the directory already exists - } - - console.log(`Downloading libvulkan...`) - const savePath = path.join(tempDir, filename) - if (!fs.existsSync(savePath)) { - await download(url, savePath) - } - - // copy to tauri resources - try { - copySync(savePath, libDir) - } catch (err) { - // Expect EEXIST error - } - - console.log('Downloads completed.') -} - -main().catch((err) => { - console.error('Error:', err) - process.exit(1) -}) diff --git a/src-tauri/icons/icon.png b/src-tauri/icons/icon.png index 1b354a2418fceb3efb272c23f443736fcc3b32d1..c16023e946407fa823feefa362903dad081c5461 100644 GIT binary patch literal 36896 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 literal 38562 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 diff --git a/src-tauri/src/core/hardware/mod.rs b/src-tauri/src/core/hardware.rs similarity index 60% rename from src-tauri/src/core/hardware/mod.rs rename to src-tauri/src/core/hardware.rs index ea2435cb0..d643d92c0 100644 --- a/src-tauri/src/core/hardware/mod.rs +++ b/src-tauri/src/core/hardware.rs @@ -1,10 +1,6 @@ -pub mod amd; -pub mod nvidia; -pub mod vulkan; - use std::sync::OnceLock; use sysinfo::System; -use tauri::{path::BaseDirectory, Manager}; +use tauri; static SYSTEM_INFO: OnceLock = OnceLock::new(); @@ -143,90 +139,12 @@ impl CpuStaticInfo { } } -// https://devicehunt.com/all-pci-vendors -pub const VENDOR_ID_AMD: u32 = 0x1002; -pub const VENDOR_ID_NVIDIA: u32 = 0x10DE; -pub const VENDOR_ID_INTEL: u32 = 0x8086; - -#[derive(Debug, Clone)] -pub enum Vendor { - AMD, - NVIDIA, - Intel, - Unknown(u32), -} - -impl serde::Serialize for Vendor { - fn serialize(&self, serializer: S) -> Result - where - S: serde::Serializer, - { - match self { - Vendor::AMD => "AMD".serialize(serializer), - Vendor::NVIDIA => "NVIDIA".serialize(serializer), - Vendor::Intel => "Intel".serialize(serializer), - Vendor::Unknown(vendor_id) => { - let formatted = format!("Unknown (vendor_id: {})", vendor_id); - serializer.serialize_str(&formatted) - } - } - } -} - -impl Vendor { - pub fn from_vendor_id(vendor_id: u32) -> Self { - match vendor_id { - VENDOR_ID_AMD => Vendor::AMD, - VENDOR_ID_NVIDIA => Vendor::NVIDIA, - VENDOR_ID_INTEL => Vendor::Intel, - _ => Vendor::Unknown(vendor_id), - } - } -} - -#[derive(Clone, Debug, serde::Serialize)] -pub struct GpuInfo { - pub name: String, - pub total_memory: u64, - pub vendor: Vendor, - pub uuid: String, - pub driver_version: String, - pub nvidia_info: Option, - pub vulkan_info: Option, -} - -impl GpuInfo { - pub fn get_usage(&self) -> GpuUsage { - match self.vendor { - Vendor::NVIDIA => self.get_usage_nvidia(), - Vendor::AMD => self.get_usage_amd(), - _ => self.get_usage_unsupported(), - } - } - - pub fn get_usage_unsupported(&self) -> GpuUsage { - GpuUsage { - uuid: self.uuid.clone(), - used_memory: 0, - total_memory: 0, - } - } -} - #[derive(serde::Serialize, Clone, Debug)] pub struct SystemInfo { cpu: CpuStaticInfo, os_type: String, os_name: String, total_memory: u64, - gpus: Vec, -} - -#[derive(serde::Serialize, Clone, Debug)] -pub struct GpuUsage { - uuid: String, - used_memory: u64, - total_memory: u64, } #[derive(serde::Serialize, Clone, Debug)] @@ -234,62 +152,15 @@ pub struct SystemUsage { cpu: f32, used_memory: u64, total_memory: u64, - gpus: Vec, -} - -fn get_jan_libvulkan_path(app: tauri::AppHandle) -> String { - let lib_name = if cfg!(target_os = "windows") { - "vulkan-1.dll" - } else if cfg!(target_os = "linux") { - "libvulkan.so" - } else { - return "".to_string(); - }; - - // NOTE: this does not work in test mode (mock app) - match app.path().resolve( - format!("resources/lib/{}", lib_name), - BaseDirectory::Resource, - ) { - Ok(lib_path) => lib_path.to_string_lossy().to_string(), - Err(_) => "".to_string(), - } } #[tauri::command] -pub fn get_system_info(app: tauri::AppHandle) -> SystemInfo { +pub fn get_system_info() -> SystemInfo { SYSTEM_INFO .get_or_init(|| { let mut system = System::new(); system.refresh_memory(); - let mut gpu_map = std::collections::HashMap::new(); - for gpu in nvidia::get_nvidia_gpus() { - gpu_map.insert(gpu.uuid.clone(), gpu); - } - - // try system vulkan first - let paths = vec!["".to_string(), get_jan_libvulkan_path(app.clone())]; - let mut vulkan_gpus = vec![]; - for path in paths { - vulkan_gpus = vulkan::get_vulkan_gpus(&path); - if !vulkan_gpus.is_empty() { - break; - } - } - - for gpu in vulkan_gpus { - match gpu_map.get_mut(&gpu.uuid) { - // for existing NVIDIA GPUs, add Vulkan info - Some(nvidia_gpu) => { - nvidia_gpu.vulkan_info = gpu.vulkan_info; - } - None => { - gpu_map.insert(gpu.uuid.clone(), gpu); - } - } - } - let os_type = if cfg!(target_os = "windows") { "windows" } else if cfg!(target_os = "macos") { @@ -306,14 +177,13 @@ pub fn get_system_info(app: tauri::AppHandle) -> SystemInf os_type: os_type.to_string(), os_name, total_memory: system.total_memory() / 1024 / 1024, // bytes to MiB - gpus: gpu_map.into_values().collect(), } }) .clone() } #[tauri::command] -pub fn get_system_usage(app: tauri::AppHandle) -> SystemUsage { +pub fn get_system_usage() -> SystemUsage { let mut system = System::new(); system.refresh_memory(); @@ -330,30 +200,22 @@ pub fn get_system_usage(app: tauri::AppHandle) -> SystemUs cpu: cpu_usage, used_memory: system.used_memory() / 1024 / 1024, // bytes to MiB, total_memory: system.total_memory() / 1024 / 1024, // bytes to MiB, - gpus: get_system_info(app.clone()) - .gpus - .iter() - .map(|gpu| gpu.get_usage()) - .collect(), } } #[cfg(test)] mod tests { use super::*; - use tauri::test::mock_app; #[test] fn test_system_info() { - let app = mock_app(); - let info = get_system_info(app.handle().clone()); + let info = get_system_info(); println!("System Static Info: {:?}", info); } #[test] fn test_system_usage() { - let app = mock_app(); - let usage = get_system_usage(app.handle().clone()); + let usage = get_system_usage(); println!("System Usage Info: {:?}", usage); } } diff --git a/src-tauri/src/core/hardware/amd.rs b/src-tauri/src/core/hardware/amd.rs deleted file mode 100644 index cbaea172d..000000000 --- a/src-tauri/src/core/hardware/amd.rs +++ /dev/null @@ -1,210 +0,0 @@ -use super::{GpuInfo, GpuUsage}; - -impl GpuInfo { - #[cfg(not(target_os = "linux"))] - #[cfg(not(target_os = "windows"))] - pub fn get_usage_amd(&self) -> GpuUsage { - self.get_usage_unsupported() - } - - #[cfg(target_os = "linux")] - pub fn get_usage_amd(&self) -> GpuUsage { - use std::fs; - use std::path::Path; - - let device_id = match &self.vulkan_info { - Some(vulkan_info) => vulkan_info.device_id, - None => { - log::error!("get_usage_amd called without Vulkan info"); - return self.get_usage_unsupported(); - } - }; - - let closure = || -> Result> { - for subdir in fs::read_dir("/sys/class/drm")? { - let device_path = subdir?.path().join("device"); - - // Check if this is an AMD GPU by looking for amdgpu directory - if !device_path - .join("driver/module/drivers/pci:amdgpu") - .exists() - { - continue; - } - - // match device_id from Vulkan info - let this_device_id_str = fs::read_to_string(device_path.join("device"))?; - let this_device_id = u32::from_str_radix( - this_device_id_str - .strip_prefix("0x") - .unwrap_or(&this_device_id_str) - .trim(), - 16, - )?; - if this_device_id != device_id { - continue; - } - - let read_mem = |path: &Path| -> u64 { - fs::read_to_string(path) - .map(|content| content.trim().parse::().unwrap_or(0)) - .unwrap_or(0) - / 1024 - / 1024 // Convert bytes to MiB - }; - return Ok(GpuUsage { - uuid: self.uuid.clone(), - total_memory: read_mem(&device_path.join("mem_info_vram_total")), - used_memory: read_mem(&device_path.join("mem_info_vram_used")), - }); - } - Err(format!("GPU not found").into()) - }; - - match closure() { - Ok(usage) => usage, - Err(e) => { - log::error!( - "Failed to get memory usage for AMD GPU {:#x}: {}", - device_id, - e - ); - self.get_usage_unsupported() - } - } - } - - #[cfg(target_os = "windows")] - pub fn get_usage_amd(&self) -> GpuUsage { - use std::collections::HashMap; - - let memory_usage_map = windows_impl::get_gpu_usage().unwrap_or_else(|_| { - log::error!("Failed to get AMD GPU memory usage"); - HashMap::new() - }); - - match memory_usage_map.get(&self.name) { - Some(&used_memory) => GpuUsage { - uuid: self.uuid.clone(), - used_memory: used_memory as u64, - total_memory: self.total_memory, - }, - None => self.get_usage_unsupported(), - } - } -} - -// TODO: refactor this into a more egonomic API -#[cfg(target_os = "windows")] -mod windows_impl { - use libc; - use libloading::{Library, Symbol}; - use std::collections::HashMap; - use std::ffi::{c_char, c_int, c_void, CStr}; - use std::mem::{self, MaybeUninit}; - use std::ptr; - - // === FFI Struct Definitions === - #[repr(C)] - #[allow(non_snake_case)] - #[derive(Debug, Copy, Clone)] - pub struct AdapterInfo { - pub iSize: c_int, - pub iAdapterIndex: c_int, - pub strUDID: [c_char; 256], - pub iBusNumber: c_int, - pub iDeviceNumber: c_int, - pub iFunctionNumber: c_int, - pub iVendorID: c_int, - pub strAdapterName: [c_char; 256], - pub strDisplayName: [c_char; 256], - pub iPresent: c_int, - pub iExist: c_int, - pub strDriverPath: [c_char; 256], - pub strDriverPathExt: [c_char; 256], - pub strPNPString: [c_char; 256], - pub iOSDisplayIndex: c_int, - } - - type ADL_MAIN_MALLOC_CALLBACK = Option *mut c_void>; - type ADL_MAIN_CONTROL_CREATE = unsafe extern "C" fn(ADL_MAIN_MALLOC_CALLBACK, c_int) -> c_int; - type ADL_MAIN_CONTROL_DESTROY = unsafe extern "C" fn() -> c_int; - type ADL_ADAPTER_NUMBEROFADAPTERS_GET = unsafe extern "C" fn(*mut c_int) -> c_int; - type ADL_ADAPTER_ADAPTERINFO_GET = unsafe extern "C" fn(*mut AdapterInfo, c_int) -> c_int; - type ADL_ADAPTER_ACTIVE_GET = unsafe extern "C" fn(c_int, *mut c_int) -> c_int; - type ADL_GET_DEDICATED_VRAM_USAGE = - unsafe extern "C" fn(*mut c_void, c_int, *mut c_int) -> c_int; - - // === ADL Memory Allocator === - unsafe extern "C" fn adl_malloc(i_size: i32) -> *mut c_void { - libc::malloc(i_size as usize) - } - - pub fn get_gpu_usage() -> Result, Box> { - unsafe { - let lib = Library::new("atiadlxx.dll").or_else(|_| Library::new("atiadlxy.dll"))?; - - let adl_main_control_create: Symbol = - lib.get(b"ADL_Main_Control_Create")?; - let adl_main_control_destroy: Symbol = - lib.get(b"ADL_Main_Control_Destroy")?; - let adl_adapter_number_of_adapters_get: Symbol = - lib.get(b"ADL_Adapter_NumberOfAdapters_Get")?; - let adl_adapter_adapter_info_get: Symbol = - lib.get(b"ADL_Adapter_AdapterInfo_Get")?; - let adl_adapter_active_get: Symbol = - lib.get(b"ADL_Adapter_Active_Get")?; - let adl_get_dedicated_vram_usage: Symbol = - lib.get(b"ADL2_Adapter_DedicatedVRAMUsage_Get")?; - - // TODO: try to put nullptr here. then we don't need direct libc dep - if adl_main_control_create(Some(adl_malloc), 1) != 0 { - return Err("ADL initialization error!".into()); - } - // NOTE: after this call, we must call ADL_Main_Control_Destroy - // whenver we encounter an error - - let mut num_adapters: c_int = 0; - if adl_adapter_number_of_adapters_get(&mut num_adapters as *mut _) != 0 { - return Err("Cannot get number of adapters".into()); - } - - let mut vram_usages = HashMap::new(); - - if num_adapters > 0 { - let mut adapter_info: Vec = - vec![MaybeUninit::zeroed().assume_init(); num_adapters as usize]; - let ret = adl_adapter_adapter_info_get( - adapter_info.as_mut_ptr(), - mem::size_of::() as i32 * num_adapters, - ); - if ret != 0 { - return Err("Cannot get adapter info".into()); - } - - for adapter in adapter_info.iter() { - let mut is_active = 0; - adl_adapter_active_get(adapter.iAdapterIndex, &mut is_active); - - if is_active != 0 { - let mut vram_mb = 0; - let _ = adl_get_dedicated_vram_usage( - ptr::null_mut(), - adapter.iAdapterIndex, - &mut vram_mb, - ); - // NOTE: adapter name might not be unique? - let name = CStr::from_ptr(adapter.strAdapterName.as_ptr()) - .to_string_lossy() - .into_owned(); - vram_usages.insert(name, vram_mb); - } - } - } - - adl_main_control_destroy(); - - Ok(vram_usages) - } - } -} diff --git a/src-tauri/src/core/hardware/nvidia.rs b/src-tauri/src/core/hardware/nvidia.rs deleted file mode 100644 index 6dced3448..000000000 --- a/src-tauri/src/core/hardware/nvidia.rs +++ /dev/null @@ -1,120 +0,0 @@ -use super::{GpuInfo, GpuUsage, Vendor}; -use nvml_wrapper::{error::NvmlError, Nvml}; -use std::sync::OnceLock; - -static NVML: OnceLock> = OnceLock::new(); - -#[derive(Debug, Clone, serde::Serialize)] -pub struct NvidiaInfo { - pub index: u32, - pub compute_capability: String, -} - -fn get_nvml() -> Option<&'static Nvml> { - NVML.get_or_init(|| { - let result = Nvml::init().or_else(|e| { - // fallback - if cfg!(target_os = "linux") { - let lib_path = std::ffi::OsStr::new("libnvidia-ml.so.1"); - Nvml::builder().lib_path(lib_path).init() - } else { - Err(e) - } - }); - - // NvmlError doesn't implement Copy, so we have to store an Option in OnceLock - match result { - Ok(nvml) => Some(nvml), - Err(e) => { - log::error!("Unable to initialize NVML: {}", e); - None - } - } - }) - .as_ref() -} - -impl GpuInfo { - pub fn get_usage_nvidia(&self) -> GpuUsage { - let index = match self.nvidia_info { - Some(ref nvidia_info) => nvidia_info.index, - None => { - log::error!("get_usage_nvidia() called on non-NVIDIA GPU"); - return self.get_usage_unsupported(); - } - }; - let closure = || -> Result { - let nvml = get_nvml().ok_or(NvmlError::Unknown)?; - let device = nvml.device_by_index(index)?; - let mem_info = device.memory_info()?; - Ok(GpuUsage { - uuid: self.uuid.clone(), - used_memory: mem_info.used / 1024 / 1024, // bytes to MiB - total_memory: mem_info.total / 1024 / 1024, // bytes to MiB - }) - }; - closure().unwrap_or_else(|e| { - log::error!("Failed to get memory usage for NVIDIA GPU {}: {}", index, e); - self.get_usage_unsupported() - }) - } -} - -pub fn get_nvidia_gpus() -> Vec { - let closure = || -> Result, NvmlError> { - let nvml = get_nvml().ok_or(NvmlError::Unknown)?; - let num_gpus = nvml.device_count()?; - let driver_version = nvml.sys_driver_version()?; - - let mut gpus = Vec::with_capacity(num_gpus as usize); - for i in 0..num_gpus { - let device = nvml.device_by_index(i)?; - gpus.push(GpuInfo { - name: device.name()?, - total_memory: device.memory_info()?.total / 1024 / 1024, // bytes to MiB - vendor: Vendor::NVIDIA, - uuid: { - let mut uuid = device.uuid()?; - if uuid.starts_with("GPU-") { - uuid = uuid[4..].to_string(); - } - uuid - }, - driver_version: driver_version.clone(), - nvidia_info: Some(NvidiaInfo { - index: i, - compute_capability: { - let cc = device.cuda_compute_capability()?; - format!("{}.{}", cc.major, cc.minor) - }, - }), - vulkan_info: None, - }); - } - - Ok(gpus) - }; - - match closure() { - Ok(gpus) => gpus, - Err(e) => { - log::error!("Failed to get NVIDIA GPUs: {}", e); - vec![] - } - } -} - -#[cfg(test)] -mod tests { - use super::*; - - #[test] - fn test_get_nvidia_gpus() { - let gpus = get_nvidia_gpus(); - for (i, gpu) in gpus.iter().enumerate() { - println!("GPU {}:", i); - println!(" {:?}", gpu); - println!(" {:?}", gpu.get_usage()); - } - } -} diff --git a/src-tauri/src/core/hardware/vulkan.rs b/src-tauri/src/core/hardware/vulkan.rs deleted file mode 100644 index cba3ed391..000000000 --- a/src-tauri/src/core/hardware/vulkan.rs +++ /dev/null @@ -1,145 +0,0 @@ -use super::{GpuInfo, Vendor}; -use ash::{vk, Entry}; - -#[derive(Debug, Clone, serde::Serialize)] -pub struct VulkanInfo { - pub index: u64, - pub device_type: String, - pub api_version: String, - pub device_id: u32, -} - -fn parse_uuid(bytes: &[u8; 16]) -> String { - format!( - "{:02x}{:02x}{:02x}{:02x}-\ - {:02x}{:02x}-\ - {:02x}{:02x}-\ - {:02x}{:02x}-\ - {:02x}{:02x}{:02x}{:02x}{:02x}{:02x}", - bytes[0], - bytes[1], - bytes[2], - bytes[3], - bytes[4], - bytes[5], - bytes[6], - bytes[7], - bytes[8], - bytes[9], - bytes[10], - bytes[11], - bytes[12], - bytes[13], - bytes[14], - bytes[15], - ) -} - -pub fn get_vulkan_gpus(lib_path: &str) -> Vec { - match get_vulkan_gpus_internal(lib_path) { - Ok(gpus) => gpus, - Err(e) => { - log::error!("Failed to get Vulkan GPUs: {:?}", e); - vec![] - } - } -} - -fn parse_c_string(buf: &[i8]) -> String { - unsafe { std::ffi::CStr::from_ptr(buf.as_ptr()) } - .to_str() - .unwrap_or_default() - .to_string() -} - -fn get_vulkan_gpus_internal(lib_path: &str) -> Result, Box> { - let entry = if lib_path.is_empty() { - unsafe { Entry::load()? } - } else { - unsafe { Entry::load_from(lib_path)? } - }; - let app_info = vk::ApplicationInfo { - api_version: vk::make_api_version(0, 1, 1, 0), - ..Default::default() - }; - let create_info = vk::InstanceCreateInfo { - p_application_info: &app_info, - ..Default::default() - }; - let instance = unsafe { entry.create_instance(&create_info, None)? }; - - let mut device_info_list = vec![]; - - for (i, device) in unsafe { instance.enumerate_physical_devices()? } - .iter() - .enumerate() - { - // create a chain of properties struct for VkPhysicalDeviceProperties2(3) - // https://registry.khronos.org/vulkan/specs/latest/man/html/VkPhysicalDeviceProperties2.html - // props2 -> driver_props -> id_props - let mut id_props = vk::PhysicalDeviceIDProperties::default(); - let mut driver_props = vk::PhysicalDeviceDriverProperties { - p_next: &mut id_props as *mut _ as *mut std::ffi::c_void, - ..Default::default() - }; - let mut props2 = vk::PhysicalDeviceProperties2 { - p_next: &mut driver_props as *mut _ as *mut std::ffi::c_void, - ..Default::default() - }; - unsafe { - instance.get_physical_device_properties2(*device, &mut props2); - } - - let props = props2.properties; - if props.device_type == vk::PhysicalDeviceType::CPU { - continue; - } - - let device_info = GpuInfo { - name: parse_c_string(&props.device_name), - total_memory: unsafe { instance.get_physical_device_memory_properties(*device) } - .memory_heaps - .iter() - .filter(|heap| heap.flags.contains(vk::MemoryHeapFlags::DEVICE_LOCAL)) - .map(|heap| heap.size / (1024 * 1024)) - .sum(), - vendor: Vendor::from_vendor_id(props.vendor_id), - uuid: parse_uuid(&id_props.device_uuid), - driver_version: parse_c_string(&driver_props.driver_info), - nvidia_info: None, - vulkan_info: Some(VulkanInfo { - index: i as u64, - device_type: format!("{:?}", props.device_type), - api_version: format!( - "{}.{}.{}", - vk::api_version_major(props.api_version), - vk::api_version_minor(props.api_version), - vk::api_version_patch(props.api_version) - ), - device_id: props.device_id, - }), - }; - device_info_list.push(device_info); - } - - unsafe { - instance.destroy_instance(None); - } - - Ok(device_info_list) -} - -#[cfg(test)] -mod tests { - use super::*; - - #[test] - fn test_get_vulkan_gpus() { - let gpus = get_vulkan_gpus(""); - for (i, gpu) in gpus.iter().enumerate() { - println!("GPU {}:", i); - println!(" {:?}", gpu); - println!(" {:?}", gpu.get_usage()); - } - } -} diff --git a/src-tauri/tauri.bundle.windows.nsis.template b/src-tauri/tauri.bundle.windows.nsis.template index 2a216c4a6..6bbff3c09 100644 --- a/src-tauri/tauri.bundle.windows.nsis.template +++ b/src-tauri/tauri.bundle.windows.nsis.template @@ -696,8 +696,6 @@ Section Install ; Copy resources CreateDirectory "$INSTDIR\resources" CreateDirectory "$INSTDIR\resources\pre-install" - SetOutPath $INSTDIR - File /a "/oname=vulkan-1.dll" "D:\a\jan\jan\src-tauri\resources\lib\vulkan-1.dll" SetOutPath "$INSTDIR\resources\pre-install" File /nonfatal /a /r "D:\a\jan\jan\src-tauri\resources\pre-install\" SetOutPath $INSTDIR diff --git a/src-tauri/tauri.linux.conf.json b/src-tauri/tauri.linux.conf.json index 48411fd3b..584e8a28c 100644 --- a/src-tauri/tauri.linux.conf.json +++ b/src-tauri/tauri.linux.conf.json @@ -10,8 +10,7 @@ }, "deb": { "files": { - "usr/bin/bun": "resources/bin/bun", - "usr/lib/Jan/resources/lib/libvulkan.so": "resources/lib/libvulkan.so" + "usr/bin/bun": "resources/bin/bun" } } } diff --git a/testRunner.js b/testRunner.js deleted file mode 100644 index 1067f05a3..000000000 --- a/testRunner.js +++ /dev/null @@ -1,19 +0,0 @@ -const jestRunner = require('jest-runner') - -class EmptyTestFileRunner extends jestRunner.default { - async runTests(tests, watcher, onStart, onResult, onFailure, options) { - const nonEmptyTests = tests.filter( - (test) => test.context.hasteFS.getSize(test.path) > 0 - ) - return super.runTests( - nonEmptyTests, - watcher, - onStart, - onResult, - onFailure, - options - ) - } -} - -module.exports = EmptyTestFileRunner diff --git a/web-app/src/hooks/useHardware.ts b/web-app/src/hooks/useHardware.ts index 183fa2490..5467aea06 100644 --- a/web-app/src/hooks/useHardware.ts +++ b/web-app/src/hooks/useHardware.ts @@ -12,30 +12,6 @@ export interface CPU { instructions?: string[] // Cortex migration: ensure instructions data ready } -export interface GPUAdditionalInfo { - compute_cap: string - driver_version: string -} - -export interface GPU { - name: string - total_memory: number - vendor: string - uuid: string - driver_version: string - activated?: boolean - nvidia_info: { - index: number - compute_capability: string - } - vulkan_info: { - index: number - device_id: number - device_type: string - api_version: string - } -} - export interface OS { name: string version: string @@ -48,7 +24,6 @@ export interface RAM { export interface HardwareData { cpu: CPU - gpus: GPU[] os_type: string os_name: string total_memory: number @@ -60,11 +35,6 @@ export interface SystemUsage { cpu: number used_memory: number total_memory: number - gpus: { - uuid: string - used_memory: number - total_memory: number - }[] } // Default values @@ -76,7 +46,6 @@ const defaultHardwareData: HardwareData = { name: '', usage: 0, }, - gpus: [], os_type: '', os_name: '', total_memory: 0, @@ -86,7 +55,6 @@ const defaultSystemUsage: SystemUsage = { cpu: 0, used_memory: 0, total_memory: 0, - gpus: [], } interface HardwareStore { @@ -96,22 +64,17 @@ interface HardwareStore { // Update functions setCPU: (cpu: CPU) => void - setGPUs: (gpus: GPU[]) => void setOS: (os: OS) => void setRAM: (ram: RAM) => void // Update entire hardware data at once setHardwareData: (data: HardwareData) => void - // Update individual GPU - updateGPU: (index: number, gpu: GPU) => void - // Update RAM available updateSystemUsage: (usage: SystemUsage) => void // GPU loading state gpuLoading: { [index: number]: boolean } - setGpuLoading: (index: number, loading: boolean) => void // Polling control pollingPaused: boolean @@ -126,13 +89,6 @@ export const useHardware = create()( systemUsage: defaultSystemUsage, gpuLoading: {}, pollingPaused: false, - setGpuLoading: (index, loading) => - set((state) => ({ - gpuLoading: { - ...state.gpuLoading, - [state.hardwareData.gpus[index].uuid]: loading, - }, - })), pausePolling: () => set({ pollingPaused: true }), resumePolling: () => set({ pollingPaused: false }), @@ -144,14 +100,6 @@ export const useHardware = create()( }, })), - setGPUs: (gpus) => - set((state) => ({ - hardwareData: { - ...state.hardwareData, - gpus, - }, - })), - setOS: (os) => set((state) => ({ hardwareData: { @@ -181,27 +129,9 @@ export const useHardware = create()( available: 0, total: 0, }, - gpus: data.gpus.map((gpu) => ({ - ...gpu, - activated: gpu.activated ?? false, - })), }, }), - updateGPU: (index, gpu) => - set((state) => { - const newGPUs = [...state.hardwareData.gpus] - if (index >= 0 && index < newGPUs.length) { - newGPUs[index] = gpu - } - return { - hardwareData: { - ...state.hardwareData, - gpus: newGPUs, - }, - } - }), - updateSystemUsage: (systemUsage) => set(() => ({ systemUsage, diff --git a/web-app/src/routes/hub/index.tsx b/web-app/src/routes/hub/index.tsx index 66e079412..08e8602ba 100644 --- a/web-app/src/routes/hub/index.tsx +++ b/web-app/src/routes/hub/index.tsx @@ -501,7 +501,7 @@ function Hub() {
- {loading ? ( + {loading && !filteredModels.length ? (
{t('hub:loadingModels')} diff --git a/web-app/src/routes/system-monitor.tsx b/web-app/src/routes/system-monitor.tsx index e7eac9dad..f09d2061b 100644 --- a/web-app/src/routes/system-monitor.tsx +++ b/web-app/src/routes/system-monitor.tsx @@ -1,6 +1,6 @@ /* eslint-disable @typescript-eslint/no-explicit-any */ import { createFileRoute } from '@tanstack/react-router' -import { useEffect, useState } from 'react' +import { useEffect } from 'react' import { useHardware } from '@/hooks/useHardware' import { Progress } from '@/components/ui/progress' import { route } from '@/constants/routes' @@ -19,12 +19,7 @@ function SystemMonitor() { const { t } = useTranslation() const { hardwareData, systemUsage, updateSystemUsage } = useHardware() - const { - devices: llamacppDevices, - fetchDevices, - } = useLlamacppDevices() - - const [isInitialized, setIsInitialized] = useState(false) + const { devices: llamacppDevices, fetchDevices } = useLlamacppDevices() useEffect(() => { // Fetch llamacpp devices @@ -46,14 +41,6 @@ function SystemMonitor() { return () => clearInterval(intervalId) }, [updateSystemUsage]) - // Initialize when hardware data and llamacpp devices are available - useEffect(() => { - if (hardwareData.gpus.length > 0 && !isInitialized) { - setIsInitialized(true) - } - }, [hardwareData.gpus.length, isInitialized]) - - // Calculate RAM usage percentage const ramUsagePercentage = toNumber(systemUsage.used_memory / hardwareData.total_memory) * 100