blob: 2f2492e692e2a82f2d41d350c2753667cef8244b [file] [log] [blame]
// Copyright 2018 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.
#pragma once
#include <ddk/driver.h>
#include <ddk/protocol/platform-device.h>
#include <ddktl/device-internal.h>
#include <zircon/assert.h>
#include "platform-device-internal.h"
// DDK platform device protocol support.
//
// :: Proxies ::
//
// ddk::PlatformDevProtocolProxy is a simple wrappers around platform_device_protocol_t. It does
// not own the pointers passed to it.
//
// :: Mixins ::
//
// ddk::PlatformDevProtocol is a mixin class that simplifies writing DDK drivers that
// implement the platform device protocol.
//
// :: Examples ::
//
// // A driver that implements a ZX_PROTOCOL_PLATFORM_DEV device.
// class PlatformDevDevice;
// using PdevDeviceType = ddk::Device<PlatformDevDevice, /* ddk mixins */>;
//
// class PlatformDevDevice : public PlatformDevDeviceType,
// public ddk::PlatformDevProtocol<PlatformDevDevice> {
// public:
// PlatformDevDevice(zx_device_t* parent)
// : PlatformDevDeviceType("my-platform-device", parent) {}
//
// zx_status_t MapMmio(uint32_t index, uint32_t cache_policy, void** out_vaddr,
// size_t* out_size, zx_paddr_t* out_paddr, zx_handle_t* out_handle);
// zx_status_t MapInterrupt(uint32_t index, uint32_t flags, zx_handle_t* out_handle);
// zx_status_t GetBti(uint32_t index, zx_handle_t* out_handle);
// zx_status_t GetDeviceInfo(pdev_device_info_t* out_info);
// zx_status_t GetBoardInfo(pdev_board_info_t* out_info);
// zx_status_t DeviceAdd(uint32_t index, device_add_args_t* args, zx_device_t** out);
// zx_status_t GetProtocol(uint32_t proto_id, uint32_t index, void* out_protocol);
// ...
// };
namespace ddk {
template <typename D>
class PlatformDevProtocol : public internal::base_protocol {
public:
PlatformDevProtocol() {
internal::CheckPlatformDevProtocolSubclass<D>();
pdev_proto_ops_.map_mmio = MapMmio;
pdev_proto_ops_.map_interrupt = MapInterrupt;
pdev_proto_ops_.get_bti = GetBti;
pdev_proto_ops_.get_device_info = GetDeviceInfo;
pdev_proto_ops_.get_board_info = GetBoardInfo;
pdev_proto_ops_.device_add = DeviceAdd;
pdev_proto_ops_.get_protocol = GetProtocol;
// Can only inherit from one base_protocol implementation.
ZX_ASSERT(ddk_proto_id_ == 0);
ddk_proto_id_ = ZX_PROTOCOL_PLATFORM_DEV;
ddk_proto_ops_ = &pdev_proto_ops_;
}
protected:
platform_device_protocol_ops_t pdev_proto_ops_ = {};
private:
static zx_status_t MapMmio(void* ctx, uint32_t index, uint32_t cache_policy, void** out_vaddr,
size_t* out_size, zx_paddr_t* out_paddr, zx_handle_t* out_handle) {
return static_cast<D*>(ctx)->MapMmio(index, cache_policy, out_vaddr, out_size, out_paddr,
out_handle);
}
static zx_status_t MapInterrupt(void* ctx, uint32_t index, uint32_t flags,
zx_handle_t* out_handle) {
return static_cast<D*>(ctx)->MapInterrupt(index, flags, out_handle);
}
static zx_status_t GetBti(void* ctx, uint32_t index, zx_handle_t* out_handle) {
return static_cast<D*>(ctx)->GetBti(index, out_handle);
}
static zx_status_t GetDeviceInfo(void* ctx, pdev_device_info_t* out_info) {
return static_cast<D*>(ctx)->GetDeviceInfo(out_info);
}
static zx_status_t GetBoardInfo(void* ctx, pdev_board_info_t* out_info) {
return static_cast<D*>(ctx)->GetBoardInfo(out_info);
}
static zx_status_t DeviceAdd(void* ctx, uint32_t index, device_add_args_t* args,
zx_device_t** out) {
return static_cast<D*>(ctx)->DeviceAdd(index, args, out);
}
static zx_status_t GetProtocol(void* ctx, uint32_t proto_id, uint32_t index,
void* out_protocol) {
return static_cast<D*>(ctx)->GetProtocol(proto_id, index, out_protocol);
}
};
class PlatformDevProtocolProxy {
public:
PlatformDevProtocolProxy(platform_device_protocol_t* proto)
: ops_(proto->ops), ctx_(proto->ctx) {}
zx_status_t MapMmio(uint32_t index, uint32_t cache_policy, void** out_vaddr,
size_t* out_size, zx_paddr_t* out_paddr, zx_handle_t* out_handle) {
return ops_->map_mmio(ctx_, index, cache_policy, out_vaddr, out_size, out_paddr,
out_handle);
}
zx_status_t MapInterrupt(uint32_t index, uint32_t flags, zx_handle_t* out_handle) {
return ops_->map_interrupt(ctx_, index, flags, out_handle);
}
zx_status_t GetBti(uint32_t index, zx_handle_t* out_handle) {
return ops_->get_bti(ctx_, index, out_handle);
}
zx_status_t GetDeviceInfo(pdev_device_info_t* out_info) {
return ops_->get_device_info(ctx_, out_info);
}
zx_status_t GetBoardInfo(pdev_board_info_t* out_info) {
return ops_->get_board_info(ctx_, out_info);
}
zx_status_t DeviceAdd(uint32_t index, device_add_args_t* args, zx_device_t** out) {
return ops_->device_add(ctx_, index, args, out);
}
zx_status_t GetProtocol(uint32_t proto_id, uint32_t index, void* out_protocol) {
return ops_->get_protocol(ctx_, proto_id, index, out_protocol);
}
private:
platform_device_protocol_ops_t* ops_;
void* ctx_;
};
} // namespace ddk