/* * Copyright (c) 2012, 2015, 2022 Jonas 'Sortie' Termansen. * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. * * kernelinfo.cpp * Lets user-space query information about the kernel. */ #include #include #include #include #include #include "net/tcp.h" #ifndef VERSIONSTR #define VERSIONSTR "unknown" #endif namespace Sortix { static const char* KernelInfo(const char* req) { if ( strcmp(req, "name") == 0 ) { return BRAND_KERNEL_NAME; } if ( strcmp(req, "version") == 0 ) { return VERSIONSTR; } if ( strcmp(req, "tagline") == 0 ) { return BRAND_RELEASE_TAGLINE; } if ( strcmp(req, "builddate") == 0 ) { return __DATE__; } if ( strcmp(req, "buildtime") == 0 ) { return __TIME__; } #if defined(__i386__) || defined(__x86_64__) if ( strcmp(req, "firmware") == 0 ) { return "bios"; } #else #warning "Name your system firmware here" #endif return NULL; } ssize_t sys_kernelinfo(const char* user_req, char* user_resp, size_t resplen) { char* req = GetStringFromUser(user_req); if ( !req ) return -1; // DEBUG if ( !strcmp(req, "virtual-usage") ) { delete[] req; size_t heap; size_t aux; size_t leaked; size_t total; KernelAddressStatistics(&heap, &aux, &leaked, &total); char str[4 * (20 + 3 * sizeof(size_t) + 1 + 8)]; snprintf(str, sizeof(str), "%20zu B heap\n%20zu B aux\n%20zu B leaked\n%20zu B total", heap, aux, leaked, total); size_t stringlen = strlen(str); if ( resplen < stringlen + 1 ) return errno = ERANGE, (ssize_t) stringlen; if ( !CopyToUser(user_resp, str, sizeof(char) * (stringlen + 1)) ) return -1; return 0; } // DEBUG if ( !strcmp(req, "tcp") ) { delete[] req; return TCP::Info(user_resp, resplen); } #ifdef __TRACE_ALLOCATION_SITES if ( !strcmp(req, "allocations") ) { delete[] req; bool exhausted = false; size_t total_needed = 0; for ( struct kernel_allocation_site* site = first_kernel_allocation_site; site; site = site->next ) { char str[256]; snprintf(str, sizeof(str), "%20zu B %zu allocations %s:%zu %s %c", site->allocation_site.current_size, site->allocation_site.allocations, site->allocation_site.file, site->allocation_site.line, site->allocation_site.function, site->next ? '\n' : '\0'); size_t stringlen = strlen(str); total_needed += stringlen; if ( exhausted ) continue; if ( resplen < stringlen ) { exhausted = true; continue; } if ( !CopyToUser(user_resp, str, sizeof(char) * stringlen) ) return -1; user_resp += stringlen; resplen -= stringlen; } if ( !exhausted && !resplen ) exhausted = true; if ( !exhausted ) { char zero = '\0'; if ( !CopyToUser(user_resp, &zero, 1) ) return -1; } if ( exhausted ) return errno = ERANGE, (ssize_t) total_needed; return 0; } #endif const char* str = KernelInfo(req); delete[] req; if ( !str ) return errno = EINVAL, -1; size_t stringlen = strlen(str); if ( resplen < stringlen + 1 ) return errno = ERANGE, (ssize_t) stringlen; if ( !CopyToUser(user_resp, str, sizeof(char) * (stringlen + 1)) ) return -1; return 0; } } // namespace Sortix