ProcFileSystem.cpp 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. #include "ProcFileSystem.h"
  2. #include "Task.h"
  3. #include <VirtualFileSystem/VirtualFileSystem.h>
  4. #include "system.h"
  5. #include "MemoryManager.h"
  6. static ProcFileSystem* s_the;
  7. ProcFileSystem& ProcFileSystem::the()
  8. {
  9. ASSERT(s_the);
  10. return *s_the;
  11. }
  12. RetainPtr<ProcFileSystem> ProcFileSystem::create()
  13. {
  14. return adopt(*new ProcFileSystem);
  15. }
  16. ProcFileSystem::ProcFileSystem()
  17. {
  18. s_the = this;
  19. }
  20. ProcFileSystem::~ProcFileSystem()
  21. {
  22. }
  23. ByteBuffer procfs$pid_vm(const Task& task)
  24. {
  25. InterruptDisabler disabler;
  26. char* buffer;
  27. auto stringImpl = StringImpl::createUninitialized(80 + task.regionCount() * 80 + 80 + task.subregionCount() * 80, buffer);
  28. memset(buffer, 0, stringImpl->length());
  29. char* ptr = buffer;
  30. ptr += ksprintf(ptr, "BEGIN END SIZE NAME\n");
  31. for (auto& region : task.regions()) {
  32. ptr += ksprintf(ptr, "%x -- %x %x %s\n",
  33. region->linearAddress.get(),
  34. region->linearAddress.offset(region->size - 1).get(),
  35. region->size,
  36. region->name.characters());
  37. }
  38. if (task.subregionCount()) {
  39. ptr += ksprintf(ptr, "\nREGION OFFSET BEGIN END SIZE NAME\n");
  40. for (auto& subregion : task.subregions()) {
  41. ptr += ksprintf(ptr, "%x %x %x -- %x %x %s\n",
  42. subregion->region->linearAddress.get(),
  43. subregion->offset,
  44. subregion->linearAddress.get(),
  45. subregion->linearAddress.offset(subregion->size - 1).get(),
  46. subregion->size,
  47. subregion->name.characters());
  48. }
  49. }
  50. *ptr = '\0';
  51. return ByteBuffer::copy((byte*)buffer, ptr - buffer);
  52. }
  53. ByteBuffer procfs$pid_stack(Task& task)
  54. {
  55. InterruptDisabler disabler;
  56. if (current != &task) {
  57. MM.unmapRegionsForTask(*current);
  58. MM.mapRegionsForTask(task);
  59. }
  60. struct RecognizedSymbol {
  61. dword address;
  62. const KSym* ksym;
  63. };
  64. Vector<RecognizedSymbol> recognizedSymbols;
  65. if (auto* eipKsym = ksymbolicate(task.tss().eip))
  66. recognizedSymbols.append({ task.tss().eip, eipKsym });
  67. for (dword* stackPtr = (dword*)task.framePtr(); task.isValidAddressForKernel(LinearAddress((dword)stackPtr)); stackPtr = (dword*)*stackPtr) {
  68. dword retaddr = stackPtr[1];
  69. if (auto* ksym = ksymbolicate(retaddr))
  70. recognizedSymbols.append({ retaddr, ksym });
  71. }
  72. size_t bytesNeeded = 0;
  73. for (auto& symbol : recognizedSymbols) {
  74. bytesNeeded += symbol.ksym->name.length() + 8 + 16;
  75. }
  76. auto buffer = ByteBuffer::createUninitialized(bytesNeeded);
  77. char* bufptr = (char*)buffer.pointer();
  78. for (auto& symbol : recognizedSymbols) {
  79. // FIXME: This doesn't actually create a file!
  80. unsigned offset = symbol.address - symbol.ksym->address;
  81. bufptr += ksprintf(bufptr, "%p %s +%u\n", symbol.address, symbol.ksym->name.characters(), offset);
  82. }
  83. buffer.trim(bufptr - (char*)buffer.pointer());
  84. if (current != &task) {
  85. MM.unmapRegionsForTask(task);
  86. MM.mapRegionsForTask(*current);
  87. }
  88. return buffer;
  89. }
  90. void ProcFileSystem::addProcess(Task& task)
  91. {
  92. ASSERT_INTERRUPTS_DISABLED();
  93. char buf[16];
  94. ksprintf(buf, "%d", task.pid());
  95. auto dir = addFile(createDirectory(buf));
  96. m_pid2inode.set(task.pid(), dir.index());
  97. addFile(createGeneratedFile("vm", [&task] { return procfs$pid_vm(task); }), dir.index());
  98. addFile(createGeneratedFile("stack", [&task] { return procfs$pid_stack(task); }), dir.index());
  99. }
  100. void ProcFileSystem::removeProcess(Task& task)
  101. {
  102. ASSERT_INTERRUPTS_DISABLED();
  103. auto pid = task.pid();
  104. auto it = m_pid2inode.find(pid);
  105. ASSERT(it != m_pid2inode.end());
  106. bool success = removeFile((*it).value);
  107. ASSERT(success);
  108. m_pid2inode.remove(pid);
  109. }
  110. ByteBuffer procfs$mm()
  111. {
  112. InterruptDisabler disabler;
  113. auto buffer = ByteBuffer::createUninitialized(1024);
  114. char* ptr = (char*)buffer.pointer();
  115. ptr += ksprintf(ptr, "Zone count: %u\n", MM.m_zones.size());
  116. ptr += ksprintf(ptr, "Free physical pages: %u\n", MM.m_freePages.size());
  117. buffer.trim(ptr - (char*)buffer.pointer());
  118. return buffer;
  119. }
  120. ByteBuffer procfs$mounts()
  121. {
  122. InterruptDisabler disabler;
  123. auto buffer = ByteBuffer::createUninitialized(VirtualFileSystem::the().mountCount() * 80);
  124. char* ptr = (char*)buffer.pointer();
  125. VirtualFileSystem::the().forEachMount([&ptr] (auto& mount) {
  126. auto& fs = mount.fileSystem();
  127. ptr += ksprintf(ptr, "%s @ ", fs.className());
  128. if (!mount.host().isValid())
  129. ptr += ksprintf(ptr, "/\n", fs.className());
  130. else
  131. ptr += ksprintf(ptr, "%u:%u\n", mount.host().fileSystemID(), mount.host().index());
  132. });
  133. buffer.trim(ptr - (char*)buffer.pointer());
  134. return buffer;
  135. }
  136. ByteBuffer procfs$kmalloc()
  137. {
  138. InterruptDisabler disabler;
  139. auto buffer = ByteBuffer::createUninitialized(128);
  140. char* ptr = (char*)buffer.pointer();
  141. ptr += ksprintf(ptr, "alloc: %u\nfree: %u\n", sum_alloc, sum_free);
  142. buffer.trim(ptr - (char*)buffer.pointer());
  143. return buffer;
  144. }
  145. ByteBuffer procfs$summary()
  146. {
  147. InterruptDisabler disabler;
  148. auto tasks = Task::allTasks();
  149. auto buffer = ByteBuffer::createUninitialized(tasks.size() * 256);
  150. char* ptr = (char*)buffer.pointer();
  151. ptr += ksprintf(ptr, "PID OWNER STATE PPID NSCHED FDS NAME\n");
  152. for (auto* task : tasks) {
  153. ptr += ksprintf(ptr, "%w %w:%w %b %w %x %w %s\n",
  154. task->pid(),
  155. task->uid(),
  156. task->gid(),
  157. task->state(),
  158. task->parentPID(),
  159. task->timesScheduled(),
  160. task->fileHandleCount(),
  161. task->name().characters());
  162. }
  163. *ptr = '\0';
  164. buffer.trim(ptr - (char*)buffer.pointer());
  165. return buffer;
  166. }
  167. bool ProcFileSystem::initialize()
  168. {
  169. SyntheticFileSystem::initialize();
  170. addFile(createGeneratedFile("mm", procfs$mm));
  171. addFile(createGeneratedFile("mounts", procfs$mounts));
  172. addFile(createGeneratedFile("kmalloc", procfs$kmalloc));
  173. addFile(createGeneratedFile("summary", procfs$summary));
  174. return true;
  175. }
  176. const char* ProcFileSystem::className() const
  177. {
  178. return "procfs";
  179. }