blob: d157ce636738a96a6766df7a919262656e6252c7 [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 <fidl/fuchsia.hardware.platform.bus/cpp/driver/fidl.h>
#include <fidl/fuchsia.hardware.platform.bus/cpp/fidl.h>
#include <lib/ddk/debug.h>
#include <lib/ddk/device.h>
#include <lib/ddk/metadata.h>
#include <lib/ddk/platform-defs.h>
#include <ddk/metadata/clock.h>
#include <soc/aml-meson/sm1-clk.h>
#include <soc/aml-s905d3/s905d3-hw.h>
#include "nelson.h"
namespace nelson {
namespace fpbus = fuchsia_hardware_platform_bus;
static const std::vector<fpbus::Mmio> clk_mmios{
{{
.base = S905D3_HIU_BASE,
.length = S905D3_HIU_LENGTH,
}},
{{
.base = S905D3_DOS_BASE,
.length = S905D3_DOS_LENGTH,
}},
{{
.base = S905D3_MSR_CLK_BASE,
.length = S905D3_MSR_CLK_LENGTH,
}},
};
constexpr clock_id_t clock_ids[] = {
{sm1_clk::CLK_RESET}, // PLACEHOLDER.
// For audio driver.
{sm1_clk::CLK_HIFI_PLL},
{sm1_clk::CLK_SYS_PLL_DIV16},
{sm1_clk::CLK_SYS_CPU_CLK_DIV16},
// For video decoder
{sm1_clk::CLK_DOS_GCLK_VDEC},
{sm1_clk::CLK_DOS},
// For GPU
{sm1_clk::CLK_GP0_PLL},
};
zx_status_t Nelson::ClkInit() {
fuchsia_hardware_clockimpl::wire::InitMetadata metadata;
metadata.steps = fidl::VectorView<fuchsia_hardware_clockimpl::wire::InitStep>::FromExternal(
clock_init_steps_.data(), clock_init_steps_.size());
const fit::result encoded_metadata = fidl::Persist(metadata);
if (!encoded_metadata.is_ok()) {
zxlogf(ERROR, "Failed to encode clock init metadata: %s",
encoded_metadata.error_value().FormatDescription().c_str());
return encoded_metadata.error_value().status();
}
const std::vector<fpbus::Metadata> clock_metadata{
{{
.type = DEVICE_METADATA_CLOCK_IDS,
.data = std::vector<uint8_t>(
reinterpret_cast<const uint8_t*>(&clock_ids),
reinterpret_cast<const uint8_t*>(&clock_ids) + sizeof(clock_ids)),
}},
{{
.type = DEVICE_METADATA_CLOCK_INIT,
.data = encoded_metadata.value(),
}},
};
const fpbus::Node clk_dev = [&clock_metadata]() {
fpbus::Node dev = {};
dev.name() = "nelson-clk";
dev.vid() = PDEV_VID_AMLOGIC;
dev.pid() = PDEV_PID_AMLOGIC_S905D3;
dev.did() = PDEV_DID_AMLOGIC_SM1_CLK;
dev.mmio() = clk_mmios;
dev.metadata() = clock_metadata;
return dev;
}();
fidl::Arena<> fidl_arena;
fdf::Arena arena('CLK_');
auto result = pbus_.buffer(arena)->NodeAdd(fidl::ToWire(fidl_arena, clk_dev));
if (!result.ok()) {
zxlogf(ERROR, "%s: NodeAdd Clk(clk_dev) request failed: %s", __func__,
result.FormatDescription().data());
return result.status();
}
if (result->is_error()) {
zxlogf(ERROR, "%s: NodeAdd Clk(clk_dev) failed: %s", __func__,
zx_status_get_string(result->error_value()));
return result->error_value();
}
return ZX_OK;
}
} // namespace nelson