1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-07-26 02:17:34 +00:00

Add a simple StringBuilder::appendf() and use it for ProcFS.

Okay, now ProcFS doesn't crash due to the crappy buffer size estimates
not really working out. This thing has dogshit performance and I will
fix that separately.
This commit is contained in:
Andreas Kling 2019-01-18 02:41:27 +01:00
parent dad58db757
commit 074edffc44
4 changed files with 83 additions and 116 deletions

View file

@ -1,4 +1,6 @@
#include "StringBuilder.h" #include "StringBuilder.h"
#include <LibC/stdarg.h>
#include "printf.cpp"
namespace AK { namespace AK {
@ -17,6 +19,16 @@ void StringBuilder::append(char ch)
m_strings.append(StringImpl::create(&ch, 1)); m_strings.append(StringImpl::create(&ch, 1));
} }
void StringBuilder::appendf(const char* fmt, ...)
{
va_list ap;
va_start(ap, fmt);
printfInternal([this] (char*&, char ch) {
append(ch);
}, nullptr, fmt, ap);
va_end(ap);
}
String StringBuilder::build() String StringBuilder::build()
{ {
auto strings = move(m_strings); auto strings = move(m_strings);

View file

@ -13,6 +13,7 @@ public:
void append(const String&); void append(const String&);
void append(String&&); void append(String&&);
void append(char); void append(char);
void appendf(const char*, ...);
String build(); String build();

View file

@ -167,7 +167,6 @@ class MemoryManager {
friend class Region; friend class Region;
friend class VMObject; friend class VMObject;
friend ByteBuffer procfs$mm(); friend ByteBuffer procfs$mm();
friend ByteBuffer procfs$regions();
public: public:
static MemoryManager& the() PURE; static MemoryManager& the() PURE;

View file

@ -6,6 +6,7 @@
#include "StdLib.h" #include "StdLib.h"
#include "i386.h" #include "i386.h"
#include "KSyms.h" #include "KSyms.h"
#include <AK/StringBuilder.h>
static ProcFS* s_the; static ProcFS* s_the;
@ -34,71 +35,59 @@ ByteBuffer procfs$pid_fds(Process& process)
ProcessInspectionHandle handle(process); ProcessInspectionHandle handle(process);
if (process.number_of_open_file_descriptors() == 0) if (process.number_of_open_file_descriptors() == 0)
return { }; return { };
char* buffer; StringBuilder builder;
auto stringImpl = StringImpl::create_uninitialized(process.number_of_open_file_descriptors() * 80, buffer);
memset(buffer, 0, stringImpl->length());
char* ptr = buffer;
for (size_t i = 0; i < process.max_open_file_descriptors(); ++i) { for (size_t i = 0; i < process.max_open_file_descriptors(); ++i) {
auto* descriptor = process.file_descriptor(i); auto* descriptor = process.file_descriptor(i);
if (!descriptor) if (!descriptor)
continue; continue;
ptr += ksprintf(ptr, "% 3u %s\n", i, descriptor->absolute_path().characters()); builder.appendf("% 3u %s\n", i, descriptor->absolute_path().characters());
} }
*ptr = '\0'; return builder.build().to_byte_buffer();
return ByteBuffer::copy((byte*)buffer, ptr - buffer);
} }
ByteBuffer procfs$pid_vm(Process& process) ByteBuffer procfs$pid_vm(Process& process)
{ {
ProcessInspectionHandle handle(process); ProcessInspectionHandle handle(process);
char* buffer; StringBuilder builder;
auto stringImpl = StringImpl::create_uninitialized(80 + process.regionCount() * 160 + 4096, buffer); builder.appendf("BEGIN END SIZE COMMIT NAME\n");
memset(buffer, 0, stringImpl->length());
char* ptr = buffer;
ptr += ksprintf(ptr, "BEGIN END SIZE COMMIT NAME\n");
for (auto& region : process.regions()) { for (auto& region : process.regions()) {
ptr += ksprintf(ptr, "%x -- %x %x %x %s\n", builder.appendf("%x -- %x %x %x %s\n",
region->linearAddress.get(), region->linearAddress.get(),
region->linearAddress.offset(region->size - 1).get(), region->linearAddress.offset(region->size - 1).get(),
region->size, region->size,
region->committed(), region->committed(),
region->name.characters()); region->name.characters());
} }
*ptr = '\0'; return builder.build().to_byte_buffer();
return ByteBuffer::copy((byte*)buffer, ptr - buffer);
} }
ByteBuffer procfs$pid_vmo(Process& process) ByteBuffer procfs$pid_vmo(Process& process)
{ {
ProcessInspectionHandle handle(process); ProcessInspectionHandle handle(process);
char* buffer; StringBuilder builder;
auto stringImpl = StringImpl::create_uninitialized(80 + process.regionCount() * 160 + 4096, buffer); builder.appendf("BEGIN END SIZE NAME\n");
memset(buffer, 0, stringImpl->length());
char* ptr = buffer;
ptr += ksprintf(ptr, "BEGIN END SIZE NAME\n");
for (auto& region : process.regions()) { for (auto& region : process.regions()) {
ptr += ksprintf(ptr, "%x -- %x %x %s\n", builder.appendf("%x -- %x %x %s\n",
region->linearAddress.get(), region->linearAddress.get(),
region->linearAddress.offset(region->size - 1).get(), region->linearAddress.offset(region->size - 1).get(),
region->size, region->size,
region->name.characters()); region->name.characters());
ptr += ksprintf(ptr, "VMO: %s \"%s\" @ %x(%u)\n", builder.appendf("VMO: %s \"%s\" @ %x(%u)\n",
region->vmo().is_anonymous() ? "anonymous" : "file-backed", region->vmo().is_anonymous() ? "anonymous" : "file-backed",
region->vmo().name().characters(), region->vmo().name().characters(),
&region->vmo(), &region->vmo(),
region->vmo().retain_count()); region->vmo().retain_count());
for (size_t i = 0; i < region->vmo().page_count(); ++i) { for (size_t i = 0; i < region->vmo().page_count(); ++i) {
auto& physical_page = region->vmo().physical_pages()[i]; auto& physical_page = region->vmo().physical_pages()[i];
ptr += ksprintf(ptr, "P%x%s(%u) ", builder.appendf("P%x%s(%u) ",
physical_page ? physical_page->paddr().get() : 0, physical_page ? physical_page->paddr().get() : 0,
region->cow_map.get(i) ? "!" : "", region->cow_map.get(i) ? "!" : "",
physical_page ? physical_page->retain_count() : 0 physical_page ? physical_page->retain_count() : 0
); );
} }
ptr += ksprintf(ptr, "\n"); builder.appendf("\n");
} }
*ptr = '\0'; return builder.build().to_byte_buffer();
return ByteBuffer::copy((byte*)buffer, ptr - buffer);
} }
ByteBuffer procfs$pid_stack(Process& process) ByteBuffer procfs$pid_stack(Process& process)
@ -117,40 +106,31 @@ ByteBuffer procfs$pid_stack(Process& process)
if (auto* ksym = ksymbolicate(retaddr)) if (auto* ksym = ksymbolicate(retaddr))
recognizedSymbols.append({ retaddr, ksym }); recognizedSymbols.append({ retaddr, ksym });
} }
size_t bytesNeeded = 0; StringBuilder builder;
for (auto& symbol : recognizedSymbols) {
bytesNeeded += strlen(symbol.ksym->name) + 8 + 16;
}
auto buffer = ByteBuffer::create_uninitialized(bytesNeeded);
char* bufptr = (char*)buffer.pointer();
for (auto& symbol : recognizedSymbols) { for (auto& symbol : recognizedSymbols) {
unsigned offset = symbol.address - symbol.ksym->address; unsigned offset = symbol.address - symbol.ksym->address;
bufptr += ksprintf(bufptr, "%p %s +%u\n", symbol.address, symbol.ksym->name, offset); builder.appendf("%p %s +%u\n", symbol.address, symbol.ksym->name, offset);
} }
buffer.trim(bufptr - (char*)buffer.pointer()); return builder.build().to_byte_buffer();
return buffer;
} }
ByteBuffer procfs$pid_regs(Process& process) ByteBuffer procfs$pid_regs(Process& process)
{ {
ProcessInspectionHandle handle(process); ProcessInspectionHandle handle(process);
auto& tss = process.tss(); auto& tss = process.tss();
auto buffer = ByteBuffer::create_uninitialized(1024); StringBuilder builder;
char* ptr = (char*)buffer.pointer(); builder.appendf("eax: %x\n", tss.eax);
ptr += ksprintf(ptr, "eax: %x\n", tss.eax); builder.appendf("ebx: %x\n", tss.ebx);
ptr += ksprintf(ptr, "ebx: %x\n", tss.ebx); builder.appendf("ecx: %x\n", tss.ecx);
ptr += ksprintf(ptr, "ecx: %x\n", tss.ecx); builder.appendf("edx: %x\n", tss.edx);
ptr += ksprintf(ptr, "edx: %x\n", tss.edx); builder.appendf("esi: %x\n", tss.esi);
ptr += ksprintf(ptr, "esi: %x\n", tss.esi); builder.appendf("edi: %x\n", tss.edi);
ptr += ksprintf(ptr, "edi: %x\n", tss.edi); builder.appendf("ebp: %x\n", tss.ebp);
ptr += ksprintf(ptr, "ebp: %x\n", tss.ebp); builder.appendf("cr3: %x\n", tss.cr3);
ptr += ksprintf(ptr, "cr3: %x\n", tss.cr3); builder.appendf("flg: %x\n", tss.eflags);
ptr += ksprintf(ptr, "flg: %x\n", tss.eflags); builder.appendf("sp: %w:%x\n", tss.ss, tss.esp);
ptr += ksprintf(ptr, "sp: %w:%x\n", tss.ss, tss.esp); builder.appendf("pc: %w:%x\n", tss.cs, tss.eip);
ptr += ksprintf(ptr, "pc: %w:%x\n", tss.cs, tss.eip); return builder.build().to_byte_buffer();
buffer.trim(ptr - (char*)buffer.pointer());
return buffer;
} }
ByteBuffer procfs$pid_exe(Process& process) ByteBuffer procfs$pid_exe(Process& process)
@ -203,66 +183,44 @@ ByteBuffer procfs$mm()
{ {
// FIXME: Implement // FIXME: Implement
InterruptDisabler disabler; InterruptDisabler disabler;
auto buffer = ByteBuffer::create_uninitialized(1024 + 80 * MM.m_vmos.size()); StringBuilder builder;
char* ptr = (char*)buffer.pointer();
for (auto* vmo : MM.m_vmos) { for (auto* vmo : MM.m_vmos) {
ptr += ksprintf(ptr, "VMO: %p %s(%u): p:%4u %s\n", builder.appendf("VMO: %p %s(%u): p:%4u %s\n",
vmo, vmo,
vmo->is_anonymous() ? "anon" : "file", vmo->is_anonymous() ? "anon" : "file",
vmo->retain_count(), vmo->retain_count(),
vmo->page_count(), vmo->page_count(),
vmo->name().characters()); vmo->name().characters());
} }
ptr += ksprintf(ptr, "VMO count: %u\n", MM.m_vmos.size()); builder.appendf("VMO count: %u\n", MM.m_vmos.size());
ptr += ksprintf(ptr, "Free physical pages: %u\n", MM.m_free_physical_pages.size()); builder.appendf("Free physical pages: %u\n", MM.m_free_physical_pages.size());
ptr += ksprintf(ptr, "Free supervisor physical pages: %u\n", MM.m_free_supervisor_physical_pages.size()); builder.appendf("Free supervisor physical pages: %u\n", MM.m_free_supervisor_physical_pages.size());
buffer.trim(ptr - (char*)buffer.pointer()); return builder.build().to_byte_buffer();
return buffer;
}
ByteBuffer procfs$regions()
{
// FIXME: Implement
InterruptDisabler disabler;
auto buffer = ByteBuffer::create_uninitialized(1024 + 80 * MM.m_regions.size());
char* ptr = (char*)buffer.pointer();
for (auto* region : MM.m_regions) {
ptr += ksprintf(ptr, "Region: %p VMO=%p %s\n",
region,
&region->vmo(),
region->name.characters());
}
ptr += ksprintf(ptr, "Region count: %u\n", MM.m_regions.size());
buffer.trim(ptr - (char*)buffer.pointer());
return buffer;
} }
ByteBuffer procfs$mounts() ByteBuffer procfs$mounts()
{ {
InterruptDisabler disabler; InterruptDisabler disabler;
auto buffer = ByteBuffer::create_uninitialized(VFS::the().mount_count() * 80); StringBuilder builder;
char* ptr = (char*)buffer.pointer(); VFS::the().for_each_mount([&builder] (auto& mount) {
VFS::the().for_each_mount([&ptr] (auto& mount) {
auto& fs = mount.guest_fs(); auto& fs = mount.guest_fs();
ptr += ksprintf(ptr, "%s @ ", fs.class_name()); builder.appendf("%s @ ", fs.class_name());
if (!mount.host().is_valid()) if (!mount.host().is_valid())
ptr += ksprintf(ptr, "/\n", fs.class_name()); builder.appendf("/\n", fs.class_name());
else else
ptr += ksprintf(ptr, "%u:%u\n", mount.host().fsid(), mount.host().index()); builder.appendf("%u:%u\n", mount.host().fsid(), mount.host().index());
}); });
buffer.trim(ptr - (char*)buffer.pointer()); return builder.build().to_byte_buffer();
return buffer;
} }
ByteBuffer procfs$cpuinfo() ByteBuffer procfs$cpuinfo()
{ {
auto buffer = ByteBuffer::create_uninitialized(256); StringBuilder builder;
char* ptr = (char*)buffer.pointer();
{ {
CPUID cpuid(0); CPUID cpuid(0);
ptr += ksprintf(ptr, "cpuid: "); builder.appendf("cpuid: ");
auto emit_dword = [&] (dword value) { auto emit_dword = [&] (dword value) {
ptr += ksprintf(ptr, "%c%c%c%c", builder.appendf("%c%c%c%c",
value & 0xff, value & 0xff,
(value >> 8) & 0xff, (value >> 8) & 0xff,
(value >> 16) & 0xff, (value >> 16) & 0xff,
@ -271,7 +229,7 @@ ByteBuffer procfs$cpuinfo()
emit_dword(cpuid.ebx()); emit_dword(cpuid.ebx());
emit_dword(cpuid.edx()); emit_dword(cpuid.edx());
emit_dword(cpuid.ecx()); emit_dword(cpuid.ecx());
ptr += ksprintf(ptr, "\n"); builder.appendf("\n");
} }
{ {
CPUID cpuid(1); CPUID cpuid(1);
@ -293,10 +251,10 @@ ByteBuffer procfs$cpuinfo()
display_family = family; display_family = family;
display_model = model; display_model = model;
} }
ptr += ksprintf(ptr, "family: %u\n", display_family); builder.appendf("family: %u\n", display_family);
ptr += ksprintf(ptr, "model: %u\n", display_model); builder.appendf("model: %u\n", display_model);
ptr += ksprintf(ptr, "stepping: %u\n", stepping); builder.appendf("stepping: %u\n", stepping);
ptr += ksprintf(ptr, "type: %u\n", type); builder.appendf("type: %u\n", type);
} }
{ {
// FIXME: Check first that this is supported by calling CPUID with eax=0x80000000 // FIXME: Check first that this is supported by calling CPUID with eax=0x80000000
@ -313,30 +271,33 @@ ByteBuffer procfs$cpuinfo()
copy_brand_string_part_to_buffer(0); copy_brand_string_part_to_buffer(0);
copy_brand_string_part_to_buffer(1); copy_brand_string_part_to_buffer(1);
copy_brand_string_part_to_buffer(2); copy_brand_string_part_to_buffer(2);
ptr += ksprintf(ptr, "brandstr: \"%s\"\n", buffer); builder.appendf("brandstr: \"%s\"\n", buffer);
} }
buffer.trim(ptr - (char*)buffer.pointer()); return builder.build().to_byte_buffer();
return buffer;
} }
ByteBuffer procfs$kmalloc() ByteBuffer procfs$kmalloc()
{ {
auto buffer = ByteBuffer::create_uninitialized(256); StringBuilder builder;
char* ptr = (char*)buffer.pointer(); builder.appendf(
ptr += ksprintf(ptr, "eternal: %u\nallocated: %u\nfree: %u\n", kmalloc_sum_eternal, sum_alloc, sum_free); "eternal: %u\n"
buffer.trim(ptr - (char*)buffer.pointer()); "allocated: %u\n"
return buffer; "free: %u\n",
kmalloc_sum_eternal,
sum_alloc,
sum_free
);
return builder.build().to_byte_buffer();
} }
ByteBuffer procfs$summary() ByteBuffer procfs$summary()
{ {
InterruptDisabler disabler; InterruptDisabler disabler;
auto processes = Process::allProcesses(); auto processes = Process::allProcesses();
auto buffer = ByteBuffer::create_uninitialized(processes.size() * 256); StringBuilder builder;
char* ptr = (char*)buffer.pointer(); builder.appendf("PID TPG PGP SID OWNER STATE PPID NSCHED FDS TTY NAME\n");
ptr += ksprintf(ptr, "PID TPG PGP SID OWNER STATE PPID NSCHED FDS TTY NAME\n");
for (auto* process : processes) { for (auto* process : processes) {
ptr += ksprintf(ptr, "% 3u % 3u % 3u % 3u % 4u % 8s % 3u % 9u % 3u % 4s %s\n", builder.appendf("% 3u % 3u % 3u % 3u % 4u % 8s % 3u % 9u % 3u % 4s %s\n",
process->pid(), process->pid(),
process->tty() ? process->tty()->pgid() : 0, process->tty() ? process->tty()->pgid() : 0,
process->pgid(), process->pgid(),
@ -349,32 +310,26 @@ ByteBuffer procfs$summary()
process->tty() ? strrchr(process->tty()->tty_name().characters(), '/') + 1 : "n/a", process->tty() ? strrchr(process->tty()->tty_name().characters(), '/') + 1 : "n/a",
process->name().characters()); process->name().characters());
} }
*ptr = '\0'; return builder.build().to_byte_buffer();
buffer.trim(ptr - (char*)buffer.pointer());
return buffer;
} }
ByteBuffer procfs$inodes() ByteBuffer procfs$inodes()
{ {
extern HashTable<Inode*>& all_inodes(); extern HashTable<Inode*>& all_inodes();
auto& vfs = VFS::the(); auto& vfs = VFS::the();
auto buffer = ByteBuffer::create_uninitialized(all_inodes().size() * 256); StringBuilder builder;
char* ptr = (char*)buffer.pointer();
for (auto it : all_inodes()) { for (auto it : all_inodes()) {
RetainPtr<Inode> inode = *it; RetainPtr<Inode> inode = *it;
String path = vfs.absolute_path(*inode); String path = vfs.absolute_path(*inode);
ptr += ksprintf(ptr, "Inode{K%x} %02u:%08u (%u) %s\n", inode.ptr(), inode->fsid(), inode->index(), inode->retain_count(), path.characters()); builder.appendf("Inode{K%x} %02u:%08u (%u) %s\n", inode.ptr(), inode->fsid(), inode->index(), inode->retain_count(), path.characters());
} }
*ptr = '\0'; return builder.build().to_byte_buffer();
buffer.trim(ptr - (char*)buffer.pointer());
return buffer;
} }
bool ProcFS::initialize() bool ProcFS::initialize()
{ {
SynthFS::initialize(); SynthFS::initialize();
add_file(create_generated_file("mm", procfs$mm)); add_file(create_generated_file("mm", procfs$mm));
add_file(create_generated_file("regions", procfs$regions));
add_file(create_generated_file("mounts", procfs$mounts)); add_file(create_generated_file("mounts", procfs$mounts));
add_file(create_generated_file("kmalloc", procfs$kmalloc)); add_file(create_generated_file("kmalloc", procfs$kmalloc));
add_file(create_generated_file("summary", procfs$summary)); add_file(create_generated_file("summary", procfs$summary));