DevicesModel.cpp 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. /*
  2. * Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "DevicesModel.h"
  7. #include <AK/JsonArray.h>
  8. #include <AK/JsonObject.h>
  9. #include <AK/JsonValue.h>
  10. #include <LibCore/DirIterator.h>
  11. #include <LibCore/File.h>
  12. #include <sys/stat.h>
  13. NonnullRefPtr<DevicesModel> DevicesModel::create()
  14. {
  15. return adopt_ref(*new DevicesModel);
  16. }
  17. DevicesModel::DevicesModel()
  18. {
  19. }
  20. DevicesModel::~DevicesModel()
  21. {
  22. }
  23. int DevicesModel::row_count(const GUI::ModelIndex&) const
  24. {
  25. return m_devices.size();
  26. }
  27. int DevicesModel::column_count(const GUI::ModelIndex&) const
  28. {
  29. return Column::__Count;
  30. }
  31. String DevicesModel::column_name(int column) const
  32. {
  33. switch (column) {
  34. case Column::Device:
  35. return "Device";
  36. case Column::Major:
  37. return "Major";
  38. case Column::Minor:
  39. return "Minor";
  40. case Column::ClassName:
  41. return "Class";
  42. case Column::Type:
  43. return "Type";
  44. default:
  45. VERIFY_NOT_REACHED();
  46. }
  47. }
  48. GUI::Variant DevicesModel::data(const GUI::ModelIndex& index, GUI::ModelRole role) const
  49. {
  50. VERIFY(is_valid(index));
  51. if (role == GUI::ModelRole::TextAlignment) {
  52. switch (index.column()) {
  53. case Column::Device:
  54. return Gfx::TextAlignment::CenterLeft;
  55. case Column::Major:
  56. return Gfx::TextAlignment::CenterRight;
  57. case Column::Minor:
  58. return Gfx::TextAlignment::CenterRight;
  59. case Column::ClassName:
  60. return Gfx::TextAlignment::CenterLeft;
  61. case Column::Type:
  62. return Gfx::TextAlignment::CenterLeft;
  63. }
  64. return {};
  65. }
  66. if (role == GUI::ModelRole::Sort) {
  67. const DeviceInfo& device = m_devices[index.row()];
  68. switch (index.column()) {
  69. case Column::Device:
  70. return device.path;
  71. case Column::Major:
  72. return device.major;
  73. case Column::Minor:
  74. return device.minor;
  75. case Column::ClassName:
  76. return device.class_name;
  77. case Column::Type:
  78. return device.type;
  79. default:
  80. VERIFY_NOT_REACHED();
  81. }
  82. }
  83. if (role == GUI::ModelRole::Display) {
  84. const DeviceInfo& device = m_devices[index.row()];
  85. switch (index.column()) {
  86. case Column::Device:
  87. return device.path;
  88. case Column::Major:
  89. return device.major;
  90. case Column::Minor:
  91. return device.minor;
  92. case Column::ClassName:
  93. return device.class_name;
  94. case Column::Type:
  95. switch (device.type) {
  96. case DeviceInfo::Type::Block:
  97. return "Block";
  98. case DeviceInfo::Type::Character:
  99. return "Character";
  100. default:
  101. VERIFY_NOT_REACHED();
  102. }
  103. default:
  104. VERIFY_NOT_REACHED();
  105. }
  106. }
  107. return {};
  108. }
  109. void DevicesModel::update()
  110. {
  111. auto proc_devices = Core::File::construct("/proc/devices");
  112. if (!proc_devices->open(Core::IODevice::OpenMode::ReadOnly))
  113. VERIFY_NOT_REACHED();
  114. auto json = JsonValue::from_string(proc_devices->read_all());
  115. VERIFY(json.has_value());
  116. m_devices.clear();
  117. json.value().as_array().for_each([this](auto& value) {
  118. JsonObject device = value.as_object();
  119. DeviceInfo device_info;
  120. device_info.major = device.get("major").to_uint();
  121. device_info.minor = device.get("minor").to_uint();
  122. device_info.class_name = device.get("class_name").to_string();
  123. String type_str = device.get("type").to_string();
  124. if (type_str == "block")
  125. device_info.type = DeviceInfo::Type::Block;
  126. else if (type_str == "character")
  127. device_info.type = DeviceInfo::Type::Character;
  128. else
  129. VERIFY_NOT_REACHED();
  130. m_devices.append(move(device_info));
  131. });
  132. auto fill_in_paths_from_dir = [this](const String& dir) {
  133. Core::DirIterator dir_iter { dir, Core::DirIterator::Flags::SkipDots };
  134. while (dir_iter.has_next()) {
  135. auto path = dir_iter.next_full_path();
  136. struct stat statbuf;
  137. if (lstat(path.characters(), &statbuf) != 0) {
  138. VERIFY_NOT_REACHED();
  139. }
  140. if (!S_ISBLK(statbuf.st_mode) && !S_ISCHR(statbuf.st_mode))
  141. continue;
  142. unsigned _major = major(statbuf.st_rdev);
  143. unsigned _minor = minor(statbuf.st_rdev);
  144. auto it = m_devices.find_if([_major, _minor](const auto& device_info) {
  145. return device_info.major == _major && device_info.minor == _minor;
  146. });
  147. if (it != m_devices.end()) {
  148. (*it).path = move(path);
  149. }
  150. }
  151. };
  152. fill_in_paths_from_dir("/dev");
  153. fill_in_paths_from_dir("/dev/pts");
  154. did_update();
  155. }