platform_linux.odin 2.8 KB

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