blob: c902975ad73fd7fc83f2589b8a70c38bc0858f2a [file] [log] [blame]
// Copyright 2016 The Fuchsia Authors
//
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file or at
// https://opensource.org/licenses/MIT
#ifndef ZIRCON_KERNEL_VM_INCLUDE_VM_VM_OBJECT_PHYSICAL_H_
#define ZIRCON_KERNEL_VM_INCLUDE_VM_VM_OBJECT_PHYSICAL_H_
#include <lib/user_copy/user_ptr.h>
#include <lib/zircon-internal/thread_annotations.h>
#include <stdint.h>
#include <zircon/listnode.h>
#include <zircon/types.h>
#include <fbl/array.h>
#include <fbl/canary.h>
#include <fbl/intrusive_double_list.h>
#include <fbl/macros.h>
#include <fbl/ref_counted.h>
#include <fbl/ref_ptr.h>
#include <kernel/mutex.h>
#include <vm/vm.h>
#include <vm/vm_object.h>
// VMO representing a physical range of memory
class VmObjectPhysical final : public VmObject, public VmDeferredDeleter<VmObjectPhysical> {
public:
static zx_status_t Create(paddr_t base, uint64_t size, fbl::RefPtr<VmObjectPhysical>* vmo);
// Define the lock retrieval functions differently depending on whether we should be returning a
// local lock instance, or the common one in the hierarchy_state_ptr. Due to the TA_RET_CAP
// statements we cannot perform |if constexpr| or equivalent indirection in the function body, and
// must have two completely different function definitions.
// In the absence of a local lock it is assumed, and enforced in vm_object_lock.h, that there is a
// shared lock in the hierarchy state. If there is both a local and a shared lock then the local
// lock is to be used for the improved lock tracking.
Lock<CriticalMutex>* lock() const override TA_RET_CAP(lock_) { return &lock_; }
Lock<CriticalMutex>& lock_ref() const override TA_RET_CAP(lock_) { return lock_; }
VmObject* self_locked() TA_REQ(lock()) TA_ASSERT(self_locked()->lock()) { return this; }
zx_status_t CreateChildSlice(uint64_t offset, uint64_t size, bool copy_name,
fbl::RefPtr<VmObject>* child_vmo) override
// This function reaches into the created child, which confuses analysis.
TA_NO_THREAD_SAFETY_ANALYSIS;
ChildType child_type() const override {
return is_slice() ? ChildType::kSlice : ChildType::kNotChild;
}
bool is_contiguous() const override { return true; }
bool is_slice() const { return is_slice_; }
uint64_t parent_user_id() const override { return parent_user_id_; }
uint64_t size_locked() const override { return size_; }
zx_status_t Lookup(uint64_t offset, uint64_t len, LookupFunction lookup_fn) override;
zx_status_t LookupContiguous(uint64_t offset, uint64_t len, paddr_t* out_paddr) override;
zx_status_t LookupContiguousLocked(uint64_t offset, uint64_t len, paddr_t* out_paddr)
TA_REQ(lock());
zx_status_t CommitRangePinned(uint64_t offset, uint64_t len, bool write) override;
zx_status_t PrefetchRange(uint64_t offset, uint64_t len) override;
void Unpin(uint64_t offset, uint64_t len) override {
// Unpin is a no-op for physical VMOs as they are always pinned.
}
void SetUserStreamSize(fbl::RefPtr<StreamSizeManager> ssm) override {
// Physical VMOs have no operations that can be told to use the user stream size, so can safely
// just ignore this request.
}
// Physical VMOs are implicitly pinned.
bool DebugIsRangePinned(uint64_t offset, uint64_t len) override { return true; }
void Dump(uint depth, bool verbose) override;
zx_status_t GetPage(uint64_t offset, uint pf_flags, list_node* alloc_list,
MultiPageRequest* page_request, vm_page_t** page, paddr_t* pa) override {
return ZX_ERR_NOT_SUPPORTED;
}
zx_status_t SetMappingCachePolicy(uint32_t cache_policy) override;
void MaybeDeadTransition() {}
private:
// private constructor (use Create())
VmObjectPhysical(paddr_t base, uint64_t size, bool is_slice_, uint64_t parent_user_id);
// private destructor, only called from refptr
~VmObjectPhysical() override;
friend fbl::RefPtr<VmObjectPhysical>;
DISALLOW_COPY_ASSIGN_AND_MOVE(VmObjectPhysical);
// members
mutable LOCK_DEP_INSTRUMENT(VmObjectPhysical, CriticalMutex, lockdep::LockFlagsNestable) lock_;
const uint64_t size_ = 0;
const paddr_t base_ = 0;
const bool is_slice_ = false;
const uint64_t parent_user_id_;
// parent pointer (may be null)
fbl::RefPtr<VmObjectPhysical> parent_ TA_GUARDED(ChildListLock::Get()) = nullptr;
};
#endif // ZIRCON_KERNEL_VM_INCLUDE_VM_VM_OBJECT_PHYSICAL_H_