blob: d11b4e4d8e3962b7346cbcd662ed6c8c5f60195f [file] [log] [blame]
// Copyright 2019 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <fuchsia/sysmem/llcpp/fidl.h>
#include <fuchsia/sysmem2/llcpp/fidl.h>
#include <lib/async-loop/cpp/loop.h>
#include <lib/async/cpp/wait.h>
#include <lib/closure-queue/closure_queue.h>
#include <lib/fidl/llcpp/heap_allocator.h>
#include <lib/inspect/cpp/inspect.h>
#include <lib/zx/bti.h>
#include <lib/zx/channel.h>
#include <limits>
#include <map>
#include <memory>
#include <unordered_set>
#include <ddk/binding.h>
#include <ddk/debug.h>
#include <ddk/device.h>
#include <ddk/driver.h>
#include <ddk/protocol/platform/device.h>
#include <ddk/protocol/sysmem.h>
#include <ddktl/device.h>
#include <ddktl/protocol/platform/device.h>
#include <ddktl/protocol/sysmem.h>
#include <fbl/vector.h>
#include <region-alloc/region-alloc.h>
#include "memory_allocator.h"
namespace sysmem_driver {
class Device;
using DdkDeviceType = ddk::Device<Device, ddk::Messageable, ddk::Unbindable>;
class Driver;
class BufferCollectionToken;
class LogicalBufferCollection;
struct Settings {
// Maximum size of a single allocation. Mainly useful for unit tests.
uint64_t max_allocation_size = UINT64_MAX;
class Device final : public DdkDeviceType,
public ddk::SysmemProtocol<Device, ddk::base_protocol>,
public MemoryAllocator::Owner {
Device(zx_device_t* parent_device, Driver* parent_driver);
static void OverrideSizeFromCommandLine(const char* name, uint64_t* memory_size);
zx_status_t Bind();
// The rest of the methods are only valid to call after Bind().
// SysmemProtocol implementation.
zx_status_t SysmemConnect(zx::channel allocator_request);
zx_status_t SysmemRegisterHeap(uint64_t heap, zx::channel heap_connection);
zx_status_t SysmemRegisterSecureMem(zx::channel tee_connection);
zx_status_t SysmemUnregisterSecureMem();
// Ddk mixin implementations.
zx_status_t DdkMessage(fidl_incoming_msg_t* msg, fidl_txn_t* txn);
void DdkUnbind(ddk::UnbindTxn txn);
void DdkRelease() { delete this; }
// MemoryAllocator::Owner implementation.
const zx::bti& bti() override;
zx_status_t CreatePhysicalVmo(uint64_t base, uint64_t size, zx::vmo* vmo_out) override;
void CheckForUnbind() override;
zx_status_t Connect(zx_handle_t allocator_request);
uint32_t pdev_device_info_vid();
uint32_t pdev_device_info_pid();
// Track/untrack the token by the koid of the server end of its FIDL
// channel. TrackToken() is only allowed after token->SerServerKoid().
// UntrackToken() is allowed even if there was never a
// token->SetServerKoid() (in which case it's a nop).
// While tracked, a token can be found with FindTokenByServerChannelKoid().
void TrackToken(BufferCollectionToken* token);
void UntrackToken(BufferCollectionToken* token);
// Finds and removes token_server_koid from unfound_token_koids_.
bool TryRemoveKoidFromUnfoundTokenList(zx_koid_t token_server_koid);
// Find the BufferCollectionToken (if any) by the koid of the server end of
// its FIDL channel.
BufferCollectionToken* FindTokenByServerChannelKoid(zx_koid_t token_server_koid);
// Get allocator for |settings|. Returns NULL if allocator is not
// registered for settings.
MemoryAllocator* GetAllocator(
const llcpp::fuchsia::sysmem2::BufferMemorySettings::Builder& settings);
// Get heap properties of a specific memory heap allocator.
// Clients should guarantee that the heap is valid and already registered
// to sysmem driver.
const llcpp::fuchsia::sysmem2::HeapProperties& GetHeapProperties(
llcpp::fuchsia::sysmem2::HeapType heap) const;
const sysmem_protocol_t* proto() const { return &in_proc_sysmem_protocol_; }
const zx_device_t* device() const { return zxdev_; }
async_dispatcher_t* dispatcher() { return loop_.dispatcher(); }
// Test hook
std::unordered_set<LogicalBufferCollection*>& logical_buffer_collections() {
return logical_buffer_collections_;
// Test hook
void AddLogicalBufferCollection(LogicalBufferCollection* collection) {
// Test hook
void RemoveLogicalBufferCollection(LogicalBufferCollection* collection) {
inspect::Node& collections_node() { return collections_node_; }
void set_settings(const Settings& settings) { settings_ = settings; }
const Settings& settings() const { return settings_; }
class SecureMemConnection {
SecureMemConnection(zx::channel connection, std::unique_ptr<async::Wait> wait_for_close);
zx_handle_t channel();
zx::channel connection_;
std::unique_ptr<async::Wait> wait_for_close_;
Driver* parent_driver_ = nullptr;
inspect::Inspector inspector_;
async::Loop loop_;
thrd_t loop_thrd_;
// Currently located at bootstrap/driver_manager:root/sysmem.
inspect::Node sysmem_root_;
inspect::Node heaps_;
inspect::Node collections_node_;
ddk::PDevProtocolClient pdev_;
zx::bti bti_;
// Initialize these to a value that won't be mistaken for a real vid or pid.
uint32_t pdev_device_info_vid_ = std::numeric_limits<uint32_t>::max();
uint32_t pdev_device_info_pid_ = std::numeric_limits<uint32_t>::max();
// In-proc sysmem interface. Essentially an in-proc version of
// fuchsia.sysmem.DriverConnector.
sysmem_protocol_t in_proc_sysmem_protocol_;
// This map allows us to look up the BufferCollectionToken by the koid of
// the server end of a BufferCollectionToken channel.
std::map<zx_koid_t, BufferCollectionToken*> tokens_by_koid_;
std::deque<zx_koid_t> unfound_token_koids_;
// Used to allocate memory to store FIDL tables for MemoryAllocator.
fidl::HeapAllocator fidl_allocator_;
// This map contains all registered memory allocators.
std::map<llcpp::fuchsia::sysmem2::HeapType, std::unique_ptr<MemoryAllocator>> allocators_;
// Some memory allocators need to be registered with properties before
// we can use them to allocate memory. We keep this map to store all the
// unregistered allocators.
std::pair<llcpp::fuchsia::sysmem2::HeapType, std::unique_ptr<MemoryAllocator>>>
// This map contains only the secure allocators, if any. The pointers are owned by allocators_.
// TODO(dustingreen): Consider unordered_map for this and some of above.
std::map<llcpp::fuchsia::sysmem2::HeapType, MemoryAllocator*> secure_allocators_;
// This flag is used to determine if the closing of the current secure mem
// connection is an error (true), or expected (false).
std::shared_ptr<std::atomic_bool> current_close_is_abort_;
// This has the connection to the securemem driver, if any. Once allocated this is supposed to
// stay allocated unless mexec is about to happen. The server end takes care of handling
// DdkSuspend() to allow mexec to work. For example, by calling secmem TA. This channel will
// close from the server end when DdkSuspend(mexec) happens, but only after
// UnregisterSecureMem().
std::unique_ptr<SecureMemConnection> secure_mem_;
std::unique_ptr<MemoryAllocator> contiguous_system_ram_allocator_;
std::unordered_set<LogicalBufferCollection*> logical_buffer_collections_;
Settings settings_;
bool waiting_for_unbind_ = false;
} // namespace sysmem_driver