blob: c4a94093830ea512b12f60ff546c9cd915cf145a [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.
#include <stdlib.h>
#include <ddk/binding.h>
#include <ddk/debug.h>
#include <ddk/device.h>
#include <ddk/driver.h>
#include <ddk/platform-defs.h>
#include <ddk/protocol/gpio.h>
#include <ddk/protocol/platform/device.h>
#define DRIVER_NAME "test-child-2"
typedef struct {
zx_device_t* zxdev;
} test_t;
static void test_release(void* ctx) {
free(ctx);
}
static zx_protocol_device_t test_device_protocol = {
.version = DEVICE_OPS_VERSION,
.release = test_release,
};
static zx_status_t test_gpio(pdev_protocol_t* pdev) {
zx_status_t status;
gpio_protocol_t gpio;
size_t actual;
status = pdev_get_protocol(pdev, ZX_PROTOCOL_GPIO, 0, &gpio, sizeof(gpio), &actual);
if (status != ZX_OK) {
zxlogf(ERROR, "%s: failed to get gpio, st = %d\n", DRIVER_NAME, status);
}
return status;
}
static zx_status_t test_bind(void* ctx, zx_device_t* parent) {
pdev_protocol_t pdev;
zx_status_t status;
zxlogf(INFO, "test_bind: %s \n", DRIVER_NAME);
status = device_get_protocol(parent, ZX_PROTOCOL_PDEV, &pdev);
if (status != ZX_OK) {
zxlogf(ERROR, "%s: could not get ZX_PROTOCOL_PDEV\n", DRIVER_NAME);
return status;
}
status = test_gpio(&pdev);
if (status != ZX_OK) {
zxlogf(ERROR, "%s: gpio test failed, st = %d\n", DRIVER_NAME, status);
}
test_t* test = calloc(1, sizeof(test_t));
if (!test) {
return ZX_ERR_NO_MEMORY;
}
device_add_args_t args = {
.version = DEVICE_ADD_ARGS_VERSION,
.name = "child-2",
.ctx = test,
.ops = &test_device_protocol,
.flags = DEVICE_ADD_NON_BINDABLE,
};
status = device_add(parent, &args, &test->zxdev);
if (status != ZX_OK) {
zxlogf(ERROR, "%s: device_add failed: %d\n", DRIVER_NAME, status);
free(test);
return status;
}
return ZX_OK;
}
static zx_driver_ops_t test_driver_ops = {
.version = DRIVER_OPS_VERSION,
.bind = test_bind,
};
ZIRCON_DRIVER_BEGIN(test_bus, test_driver_ops, "zircon", "0.1", 4)
BI_ABORT_IF(NE, BIND_PROTOCOL, ZX_PROTOCOL_PDEV),
BI_ABORT_IF(NE, BIND_PLATFORM_DEV_VID, PDEV_VID_TEST),
BI_ABORT_IF(NE, BIND_PLATFORM_DEV_PID, PDEV_PID_PBUS_TEST),
BI_MATCH_IF(EQ, BIND_PLATFORM_DEV_DID, PDEV_DID_TEST_CHILD_2),
ZIRCON_DRIVER_END(test_bus)