platform_linux.odin 2.9 KB

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