|  | // 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. | 
|  |  | 
|  | #ifndef SRC_DEVICES_INTERNAL_DRIVERS_FRAGMENT_FRAGMENT_H_ | 
|  | #define SRC_DEVICES_INTERNAL_DRIVERS_FRAGMENT_FRAGMENT_H_ | 
|  |  | 
|  | #include <lib/sync/completion.h> | 
|  | #include <lib/zx/channel.h> | 
|  |  | 
|  | #include <ddk/binding.h> | 
|  | #include <ddk/device.h> | 
|  | #include <ddk/driver.h> | 
|  | #include <ddktl/device.h> | 
|  | #include <ddktl/protocol/amlogiccanvas.h> | 
|  | #include <ddktl/protocol/buttons.h> | 
|  | #include <ddktl/protocol/camera/sensor.h> | 
|  | #include <ddktl/protocol/camerasensor.h> | 
|  | #include <ddktl/protocol/clock.h> | 
|  | #include <ddktl/protocol/codec.h> | 
|  | #include <ddktl/protocol/dsi.h> | 
|  | #include <ddktl/protocol/dsiimpl.h> | 
|  | #include <ddktl/protocol/ethernet/board.h> | 
|  | #include <ddktl/protocol/gdc.h> | 
|  | #include <ddktl/protocol/ge2d.h> | 
|  | #include <ddktl/protocol/goldfish/addressspace.h> | 
|  | #include <ddktl/protocol/goldfish/pipe.h> | 
|  | #include <ddktl/protocol/gpio.h> | 
|  | #include <ddktl/protocol/i2c.h> | 
|  | #include <ddktl/protocol/isp.h> | 
|  | #include <ddktl/protocol/mipicsi.h> | 
|  | #include <ddktl/protocol/platform/device.h> | 
|  | #include <ddktl/protocol/power.h> | 
|  | #include <ddktl/protocol/powerimpl.h> | 
|  | #include <ddktl/protocol/pwm.h> | 
|  | #include <ddktl/protocol/rpmb.h> | 
|  | #include <ddktl/protocol/scpi.h> | 
|  | #include <ddktl/protocol/sdio.h> | 
|  | #include <ddktl/protocol/shareddma.h> | 
|  | #include <ddktl/protocol/spi.h> | 
|  | #include <ddktl/protocol/sysmem.h> | 
|  | #include <ddktl/protocol/tee.h> | 
|  | #include <ddktl/protocol/thermal.h> | 
|  | #include <ddktl/protocol/usb/modeswitch.h> | 
|  | #include <ddktl/protocol/usb/phy.h> | 
|  | #include <ddktl/protocol/vreg.h> | 
|  |  | 
|  | namespace fragment { | 
|  |  | 
|  | template <typename ProtoClientType, typename ProtoType> | 
|  | class ProtocolClient { | 
|  | public: | 
|  | ProtocolClient() { parent_ = nullptr; } | 
|  | ProtocolClient(zx_device_t* parent, uint32_t proto_id); | 
|  | ProtoClientType& proto_client() { return proto_client_; } | 
|  | ~ProtocolClient() { | 
|  | if (is_session_) { | 
|  | ZX_DEBUG_ASSERT(device_close_protocol_session_multibindable(parent_, proto_.ctx) == ZX_OK); | 
|  | } | 
|  | } | 
|  |  | 
|  | private: | 
|  | bool is_session_ = false; | 
|  | ProtoType proto_; | 
|  | ProtoClientType proto_client_; | 
|  | zx_device_t* parent_; | 
|  | }; | 
|  |  | 
|  | class Fragment; | 
|  | using FragmentBase = ddk::Device<Fragment, ddk::Rxrpcable, ddk::Unbindable, ddk::GetProtocolable>; | 
|  |  | 
|  | class Fragment : public FragmentBase { | 
|  | public: | 
|  | explicit Fragment(zx_device_t* parent) | 
|  | : FragmentBase(parent), | 
|  | canvas_client_(parent, ZX_PROTOCOL_AMLOGIC_CANVAS), | 
|  | buttons_client_(parent, ZX_PROTOCOL_BUTTONS), | 
|  | clock_client_(parent, ZX_PROTOCOL_CLOCK), | 
|  | eth_board_client_(parent, ZX_PROTOCOL_ETH_BOARD), | 
|  | goldfish_address_space_client_(parent, ZX_PROTOCOL_GOLDFISH_ADDRESS_SPACE), | 
|  | goldfish_pipe_client_(parent, ZX_PROTOCOL_GOLDFISH_PIPE), | 
|  | gpio_client_(parent, ZX_PROTOCOL_GPIO), | 
|  | i2c_client_(parent, ZX_PROTOCOL_I2C), | 
|  | codec_client_(parent, ZX_PROTOCOL_CODEC), | 
|  | pdev_client_(parent, ZX_PROTOCOL_PDEV), | 
|  | power_client_(parent, ZX_PROTOCOL_POWER), | 
|  | pwm_client_(parent, ZX_PROTOCOL_PWM), | 
|  | spi_client_(parent, ZX_PROTOCOL_SPI), | 
|  | sysmem_client_(parent, ZX_PROTOCOL_SYSMEM), | 
|  | tee_client_(parent, ZX_PROTOCOL_TEE), | 
|  | ums_client_(parent, ZX_PROTOCOL_USB_MODE_SWITCH), | 
|  | power_impl_client_(parent, ZX_PROTOCOL_POWER_IMPL), | 
|  | dsi_impl_client_(parent, ZX_PROTOCOL_DSI_IMPL), | 
|  | sdio_client_(parent, ZX_PROTOCOL_SDIO), | 
|  | thermal_client_(parent, ZX_PROTOCOL_THERMAL), | 
|  | isp_client_(parent, ZX_PROTOCOL_ISP), | 
|  | shared_dma_client_(parent, ZX_PROTOCOL_SHARED_DMA), | 
|  | usb_phy_client_(parent, ZX_PROTOCOL_USB_PHY), | 
|  | mipi_csi_client_(parent, ZX_PROTOCOL_MIPI_CSI), | 
|  | camera_sensor_client_(parent, ZX_PROTOCOL_CAMERA_SENSOR), | 
|  | camera_sensor2_client_(parent, ZX_PROTOCOL_CAMERA_SENSOR2), | 
|  | gdc_client_(parent, ZX_PROTOCOL_GDC), | 
|  | ge2d_client_(parent, ZX_PROTOCOL_GE2D), | 
|  | scpi_client_(parent, ZX_PROTOCOL_SCPI), | 
|  | rpmb_client_(parent, ZX_PROTOCOL_RPMB), | 
|  | vreg_client_(parent, ZX_PROTOCOL_VREG), | 
|  | dsi_client_(parent, ZX_PROTOCOL_DSI) {} | 
|  |  | 
|  | static zx_status_t Bind(void* ctx, zx_device_t* parent); | 
|  |  | 
|  | zx_status_t DdkRxrpc(zx_handle_t channel); | 
|  | void DdkUnbind(ddk::UnbindTxn txn); | 
|  | void DdkRelease(); | 
|  | zx_status_t DdkGetProtocol(uint32_t proto_id, void* out_protocol); | 
|  |  | 
|  | private: | 
|  | struct I2cTransactContext { | 
|  | sync_completion_t completion; | 
|  | void* read_buf; | 
|  | size_t read_length; | 
|  | zx_status_t result; | 
|  | }; | 
|  | struct CodecTransactContext { | 
|  | sync_completion_t completion; | 
|  | zx_status_t status; | 
|  | void* buffer; | 
|  | size_t size; | 
|  | }; | 
|  | zx_status_t RpcCanvas(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcButtons(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, | 
|  | uint32_t req_handle_count, zx::handle* resp_handles, | 
|  | uint32_t* resp_handle_count); | 
|  | zx_status_t RpcClock(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcEthBoard(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, | 
|  | uint32_t req_handle_count, zx::handle* resp_handles, | 
|  | uint32_t* resp_handle_count); | 
|  | zx_status_t RpcGoldfishAddressSpace(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, | 
|  | uint32_t req_handle_count, zx::handle* resp_handles, | 
|  | uint32_t* resp_handle_count); | 
|  | zx_status_t RpcGoldfishPipe(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, | 
|  | uint32_t req_handle_count, zx::handle* resp_handles, | 
|  | uint32_t* resp_handle_count); | 
|  | zx_status_t RpcGpio(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcI2c(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcPdev(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcPower(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcPwm(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcSpi(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcSysmem(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcTee(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcUms(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcCodec(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcRpmb(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcVreg(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  | zx_status_t RpcDsi(const uint8_t* req_buf, uint32_t req_size, uint8_t* resp_buf, | 
|  | uint32_t* out_resp_size, zx::handle* req_handles, uint32_t req_handle_count, | 
|  | zx::handle* resp_handles, uint32_t* resp_handle_count); | 
|  |  | 
|  | static void I2cTransactCallback(void* cookie, zx_status_t status, const i2c_op_t* op_list, | 
|  | size_t op_count); | 
|  |  | 
|  | static void CodecTransactCallback(void* cookie, zx_status_t status, | 
|  | const dai_supported_formats_t* formats_list, | 
|  | size_t formats_count); | 
|  | ProtocolClient<ddk::AmlogicCanvasProtocolClient, amlogic_canvas_protocol_t> canvas_client_; | 
|  | ProtocolClient<ddk::ButtonsProtocolClient, buttons_protocol_t> buttons_client_; | 
|  | ProtocolClient<ddk::ClockProtocolClient, clock_protocol_t> clock_client_; | 
|  | ProtocolClient<ddk::EthBoardProtocolClient, eth_board_protocol_t> eth_board_client_; | 
|  | ProtocolClient<ddk::GoldfishAddressSpaceProtocolClient, goldfish_address_space_protocol_t> | 
|  | goldfish_address_space_client_; | 
|  | ProtocolClient<ddk::GoldfishPipeProtocolClient, goldfish_pipe_protocol_t> goldfish_pipe_client_; | 
|  | ProtocolClient<ddk::GpioProtocolClient, gpio_protocol_t> gpio_client_; | 
|  | ProtocolClient<ddk::I2cProtocolClient, i2c_protocol_t> i2c_client_; | 
|  | ProtocolClient<ddk::CodecProtocolClient, codec_protocol_t> codec_client_; | 
|  | ProtocolClient<ddk::PDevProtocolClient, pdev_protocol_t> pdev_client_; | 
|  | ProtocolClient<ddk::PowerProtocolClient, power_protocol_t> power_client_; | 
|  | ProtocolClient<ddk::PwmProtocolClient, pwm_protocol_t> pwm_client_; | 
|  | ProtocolClient<ddk::SpiProtocolClient, spi_protocol_t> spi_client_; | 
|  | ProtocolClient<ddk::SysmemProtocolClient, sysmem_protocol_t> sysmem_client_; | 
|  | ProtocolClient<ddk::TeeProtocolClient, tee_protocol_t> tee_client_; | 
|  | ProtocolClient<ddk::UsbModeSwitchProtocolClient, usb_mode_switch_protocol_t> ums_client_; | 
|  | ProtocolClient<ddk::PowerImplProtocolClient, power_impl_protocol_t> power_impl_client_; | 
|  | ProtocolClient<ddk::DsiImplProtocolClient, dsi_impl_protocol_t> dsi_impl_client_; | 
|  | ProtocolClient<ddk::SdioProtocolClient, sdio_protocol_t> sdio_client_; | 
|  | ProtocolClient<ddk::ThermalProtocolClient, thermal_protocol_t> thermal_client_; | 
|  | ProtocolClient<ddk::IspProtocolClient, isp_protocol_t> isp_client_; | 
|  | ProtocolClient<ddk::SharedDmaProtocolClient, shared_dma_protocol_t> shared_dma_client_; | 
|  | ProtocolClient<ddk::UsbPhyProtocolClient, usb_phy_protocol_t> usb_phy_client_; | 
|  | ProtocolClient<ddk::MipiCsiProtocolClient, mipi_csi_protocol_t> mipi_csi_client_; | 
|  | ProtocolClient<ddk::CameraSensorProtocolClient, camera_sensor_protocol_t> camera_sensor_client_; | 
|  | ProtocolClient<ddk::CameraSensor2ProtocolClient, camera_sensor2_protocol_t> | 
|  | camera_sensor2_client_; | 
|  | ProtocolClient<ddk::GdcProtocolClient, gdc_protocol_t> gdc_client_; | 
|  | ProtocolClient<ddk::Ge2dProtocolClient, ge2d_protocol_t> ge2d_client_; | 
|  | ProtocolClient<ddk::ScpiProtocolClient, scpi_protocol_t> scpi_client_; | 
|  | ProtocolClient<ddk::RpmbProtocolClient, rpmb_protocol_t> rpmb_client_; | 
|  | ProtocolClient<ddk::VregProtocolClient, vreg_protocol_t> vreg_client_; | 
|  | ProtocolClient<ddk::DsiProtocolClient, dsi_protocol_t> dsi_client_; | 
|  | }; | 
|  |  | 
|  | }  // namespace fragment | 
|  |  | 
|  | #endif  // SRC_DEVICES_INTERNAL_DRIVERS_FRAGMENT_FRAGMENT_H_ |