/* * Copyright (c) 2021, Patrick Meyer * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include namespace Kernel { HashMap* KCOVDevice::proc_instance; HashMap* KCOVDevice::thread_instance; UNMAP_AFTER_INIT NonnullLockRefPtr KCOVDevice::must_create() { auto kcov_device_or_error = DeviceManagement::try_create_device(); // FIXME: Find a way to propagate errors VERIFY(!kcov_device_or_error.is_error()); return kcov_device_or_error.release_value(); } UNMAP_AFTER_INIT KCOVDevice::KCOVDevice() : CharacterDevice(30, 0) { proc_instance = new HashMap(); thread_instance = new HashMap(); dbgln("KCOVDevice created"); } void KCOVDevice::free_thread() { auto thread = Thread::current(); auto tid = thread->tid(); auto maybe_kcov_instance = thread_instance->get(tid); if (!maybe_kcov_instance.has_value()) return; auto kcov_instance = maybe_kcov_instance.value(); VERIFY(kcov_instance->state() == KCOVInstance::TRACING); kcov_instance->set_state(KCOVInstance::OPENED); thread_instance->remove(tid); } void KCOVDevice::free_process() { auto pid = Process::current().pid(); auto maybe_kcov_instance = proc_instance->get(pid); if (!maybe_kcov_instance.has_value()) return; auto kcov_instance = maybe_kcov_instance.value(); VERIFY(kcov_instance->state() == KCOVInstance::OPENED); kcov_instance->set_state(KCOVInstance::UNUSED); proc_instance->remove(pid); delete kcov_instance; } ErrorOr> KCOVDevice::open(int options) { auto pid = Process::current().pid(); if (proc_instance->get(pid).has_value()) return EBUSY; // This process already open()ed the kcov device auto kcov_instance = new KCOVInstance(pid); kcov_instance->set_state(KCOVInstance::OPENED); proc_instance->set(pid, kcov_instance); return Device::open(options); } ErrorOr KCOVDevice::ioctl(OpenFileDescription&, unsigned request, Userspace arg) { auto thread = Thread::current(); auto tid = thread->tid(); auto pid = thread->pid(); auto maybe_kcov_instance = proc_instance->get(pid); if (!maybe_kcov_instance.has_value()) return ENXIO; // This proc hasn't opened the kcov dev yet auto kcov_instance = maybe_kcov_instance.value(); SpinlockLocker locker(kcov_instance->spinlock()); switch (request) { case KCOV_SETBUFSIZE: if (kcov_instance->state() >= KCOVInstance::TRACING) return EBUSY; return kcov_instance->buffer_allocate((FlatPtr)arg.unsafe_userspace_ptr()); case KCOV_ENABLE: if (kcov_instance->state() >= KCOVInstance::TRACING) return EBUSY; if (!kcov_instance->has_buffer()) return ENOBUFS; VERIFY(kcov_instance->state() == KCOVInstance::OPENED); kcov_instance->set_state(KCOVInstance::TRACING); thread_instance->set(tid, kcov_instance); return {}; case KCOV_DISABLE: { auto maybe_kcov_instance = thread_instance->get(tid); if (!maybe_kcov_instance.has_value()) return ENOENT; VERIFY(kcov_instance->state() == KCOVInstance::TRACING); kcov_instance->set_state(KCOVInstance::OPENED); thread_instance->remove(tid); return {}; } default: return EINVAL; } } ErrorOr> KCOVDevice::vmobject_for_mmap(Process& process, Memory::VirtualRange const&, u64&, bool) { auto pid = process.pid(); auto maybe_kcov_instance = proc_instance->get(pid); VERIFY(maybe_kcov_instance.has_value()); // Should happen on fd open() auto kcov_instance = maybe_kcov_instance.value(); if (!kcov_instance->vmobject()) return ENOBUFS; // mmaped, before KCOV_SETBUFSIZE return *kcov_instance->vmobject(); } }