platform_linux.odin 2.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. package sysinfo
  2. import "base:intrinsics"
  3. import "core:strconv"
  4. import "core:strings"
  5. import "core:sys/linux"
  6. @(private)
  7. version_string_buf: [1024]u8
  8. @(init, private)
  9. init_os_version :: proc () {
  10. os_version.platform = .Linux
  11. b := strings.builder_from_bytes(version_string_buf[:])
  12. // Try to parse `/etc/os-release` for `PRETTY_NAME="Ubuntu 20.04.3 LTS`
  13. {
  14. fd, errno := linux.open("/etc/os-release", {})
  15. assert(errno == .NONE, "Failed to read /etc/os-release")
  16. defer {
  17. cerrno := linux.close(fd)
  18. assert(cerrno == .NONE, "Failed to close the file descriptor")
  19. }
  20. os_release_buf: [2048]u8
  21. n, read_errno := linux.read(fd, os_release_buf[:])
  22. assert(read_errno == .NONE, "Failed to read data from /etc/os-release")
  23. release := string(os_release_buf[:n])
  24. // Search the line in the file until we find "PRETTY_NAME="
  25. NEEDLE :: "PRETTY_NAME=\""
  26. _, _, post := strings.partition(release, NEEDLE)
  27. if len(post) > 0 {
  28. end := strings.index_any(post, "\"\n")
  29. if end > -1 && post[end] == '"' {
  30. strings.write_string(&b, post[:end])
  31. }
  32. }
  33. if strings.builder_len(b) == 0 {
  34. strings.write_string(&b, "Unknown Linux Distro")
  35. }
  36. }
  37. // Grab kernel info using `uname()` syscall, https://linux.die.net/man/2/uname
  38. uts: linux.UTS_Name
  39. uname_errno := linux.uname(&uts)
  40. assert(uname_errno == .NONE, "This should never happen!")
  41. // Append the system name (typically "Linux") and kernel release (looks like 6.5.2-arch1-1)
  42. strings.write_string(&b, ", ")
  43. strings.write_string(&b, string(cstring(&uts.sysname[0])))
  44. strings.write_rune(&b, ' ')
  45. release_i := strings.builder_len(b)
  46. strings.write_string(&b, string(cstring(&uts.release[0])))
  47. release_str := string(b.buf[release_i:])
  48. os_version.as_string = strings.to_string(b)
  49. // Parse the Linux version out of the release string
  50. {
  51. version_num, _, version_suffix := strings.partition(release_str, "-")
  52. os_version.version = version_suffix
  53. i: int
  54. for part in strings.split_iterator(&version_num, ".") {
  55. defer i += 1
  56. dst: ^int
  57. switch i {
  58. case 0: dst = &os_version.major
  59. case 1: dst = &os_version.minor
  60. case 2: dst = &os_version.patch
  61. case: break
  62. }
  63. num, ok := strconv.parse_int(part)
  64. if !ok { break }
  65. dst^ = num
  66. }
  67. }
  68. }
  69. @(init, private)
  70. init_ram :: proc() {
  71. // Retrieve RAM info using `sysinfo`
  72. sys_info: linux.Sys_Info
  73. errno := linux.sysinfo(&sys_info)
  74. assert(errno == .NONE, "Good luck to whoever's debugging this, something's seriously cucked up!")
  75. ram = RAM{
  76. total_ram = int(sys_info.totalram) * int(sys_info.mem_unit),
  77. free_ram = int(sys_info.freeram) * int(sys_info.mem_unit),
  78. total_swap = int(sys_info.totalswap) * int(sys_info.mem_unit),
  79. free_swap = int(sys_info.freeswap) * int(sys_info.mem_unit),
  80. }
  81. }