blob: 56f3a0eb802bb53048c7f43979efab1f06040065 [file] [log] [blame]
// Copyright 2022 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 <fidl/fuchsia.examples.gizmo/cpp/wire.h>
#include <lib/ddk/binding_driver.h>
#include <lib/ddk/debug.h>
#include <lib/ddk/device.h>
#include <ddktl/device.h>
namespace zircon_transport {
class ZirconClientDevice;
using DeviceType = ddk::Device<ZirconClientDevice, ddk::Initializable>;
class ZirconClientDevice : public DeviceType {
public:
static zx_status_t Create(void* ctx, zx_device_t* parent);
explicit ZirconClientDevice(zx_device_t* parent) : DeviceType(parent) {}
virtual ~ZirconClientDevice() = default;
zx_status_t Bind();
// Device protocol implementation.
void DdkInit(ddk::InitTxn txn) { txn.Reply(ZX_OK); }
void DdkRelease() { delete this; }
};
// static
zx_status_t ZirconClientDevice::Create(void* ctx, zx_device_t* parent) {
auto device = std::make_unique<zircon_transport::ZirconClientDevice>(parent);
auto status = device->Bind();
if (status == ZX_OK) {
// Driver framework now owns device.
[[maybe_unused]] auto* dev = device.release();
}
return status;
}
zx_status_t ZirconClientDevice::Bind() {
auto client_end = DdkConnectFidlProtocol<fuchsia_examples_gizmo::Service::Device>();
if (client_end.is_error()) {
zxlogf(ERROR, "Failed to connect fidl protocol");
return client_end.status_value();
}
auto client = fidl::WireSyncClient(std::move(*client_end));
auto hardware_result = client->GetHardwareId();
if (!hardware_result.ok()) {
zxlogf(ERROR, "Failed to request hardware ID: %s", hardware_result.status_string());
return hardware_result.status();
} else if (hardware_result->is_error()) {
zxlogf(ERROR, "Hardware ID request returned an error: %d", hardware_result->error_value());
return hardware_result->error_value();
}
zxlogf(INFO, "Transport client hardware: %X", hardware_result.value().value()->response);
auto firmware_result = client->GetFirmwareVersion();
if (!firmware_result.ok()) {
zxlogf(ERROR, "Failed to request firmware version: %s", firmware_result.status_string());
return firmware_result.status();
} else if (firmware_result->is_error()) {
zxlogf(ERROR, "Firmware version request returned an error: %d", firmware_result->error_value());
return firmware_result->error_value();
}
zxlogf(INFO, "Transport client firmware: %d.%d", firmware_result.value().value()->major,
firmware_result.value().value()->minor);
return DdkAdd(ddk::DeviceAddArgs("test").set_flags(DEVICE_ADD_NON_BINDABLE));
}
} // namespace zircon_transport
static zx_driver_ops_t zircon_client_driver_ops = []() -> zx_driver_ops_t {
zx_driver_ops_t ops = {};
ops.version = DRIVER_OPS_VERSION;
ops.bind = zircon_transport::ZirconClientDevice::Create;
return ops;
}();
ZIRCON_DRIVER(ZirconClientDevice, zircon_client_driver_ops, "zircon", "0.1");