uboot/drivers/misc/tegra186_bpmp.c
<<
>>
Prefs
   1// SPDX-License-Identifier: GPL-2.0
   2/*
   3 * Copyright (c) 2016, NVIDIA CORPORATION.
   4 */
   5
   6#include <common.h>
   7#include <dm.h>
   8#include <log.h>
   9#include <malloc.h>
  10#include <time.h>
  11#include <dm/lists.h>
  12#include <dm/root.h>
  13#include <mailbox.h>
  14#include <misc.h>
  15#include <asm/arch-tegra/bpmp_abi.h>
  16#include <asm/arch-tegra/ivc.h>
  17#include <linux/bitops.h>
  18#include <linux/err.h>
  19
  20#define BPMP_IVC_FRAME_COUNT 1
  21#define BPMP_IVC_FRAME_SIZE 128
  22
  23#define BPMP_FLAG_DO_ACK        BIT(0)
  24#define BPMP_FLAG_RING_DOORBELL BIT(1)
  25
  26DECLARE_GLOBAL_DATA_PTR;
  27
  28struct tegra186_bpmp {
  29        struct mbox_chan mbox;
  30        struct tegra_ivc ivc;
  31};
  32
  33static int tegra186_bpmp_call(struct udevice *dev, int mrq, void *tx_msg,
  34                              int tx_size, void *rx_msg, int rx_size)
  35{
  36        struct tegra186_bpmp *priv = dev_get_priv(dev);
  37        int ret, err;
  38        void *ivc_frame;
  39        struct mrq_request *req;
  40        struct mrq_response *resp;
  41        ulong start_time;
  42
  43        debug("%s(dev=%p, mrq=%u, tx_msg=%p, tx_size=%d, rx_msg=%p, rx_size=%d) (priv=%p)\n",
  44              __func__, dev, mrq, tx_msg, tx_size, rx_msg, rx_size, priv);
  45
  46        if ((tx_size > BPMP_IVC_FRAME_SIZE) || (rx_size > BPMP_IVC_FRAME_SIZE))
  47                return -EINVAL;
  48
  49        ret = tegra_ivc_write_get_next_frame(&priv->ivc, &ivc_frame);
  50        if (ret) {
  51                pr_err("tegra_ivc_write_get_next_frame() failed: %d\n", ret);
  52                return ret;
  53        }
  54
  55        req = ivc_frame;
  56        req->mrq = mrq;
  57        req->flags = BPMP_FLAG_DO_ACK | BPMP_FLAG_RING_DOORBELL;
  58        memcpy(req + 1, tx_msg, tx_size);
  59
  60        ret = tegra_ivc_write_advance(&priv->ivc);
  61        if (ret) {
  62                pr_err("tegra_ivc_write_advance() failed: %d\n", ret);
  63                return ret;
  64        }
  65
  66        start_time = timer_get_us();
  67        for (;;) {
  68                ret = tegra_ivc_channel_notified(&priv->ivc);
  69                if (ret) {
  70                        pr_err("tegra_ivc_channel_notified() failed: %d\n", ret);
  71                        return ret;
  72                }
  73
  74                ret = tegra_ivc_read_get_next_frame(&priv->ivc, &ivc_frame);
  75                if (!ret)
  76                        break;
  77
  78                /* Timeout 20ms; roughly 10x current max observed duration */
  79                if ((timer_get_us() - start_time) > 20 * 1000) {
  80                        pr_err("tegra_ivc_read_get_next_frame() timed out (%d)\n",
  81                              ret);
  82                        return -ETIMEDOUT;
  83                }
  84        }
  85
  86        resp = ivc_frame;
  87        err = resp->err;
  88        if (!err && rx_msg && rx_size)
  89                memcpy(rx_msg, resp + 1, rx_size);
  90
  91        ret = tegra_ivc_read_advance(&priv->ivc);
  92        if (ret) {
  93                pr_err("tegra_ivc_write_advance() failed: %d\n", ret);
  94                return ret;
  95        }
  96
  97        if (err) {
  98                pr_err("BPMP responded with error %d\n", err);
  99                /* err isn't a U-Boot error code, so don't that */
 100                return -EIO;
 101        }
 102
 103        return rx_size;
 104}
 105
 106/**
 107 * The BPMP exposes multiple different services. We create a sub-device for
 108 * each separate type of service, since each device must be of the appropriate
 109 * UCLASS.
 110 */
 111static int tegra186_bpmp_bind(struct udevice *dev)
 112{
 113        int ret;
 114        struct udevice *child;
 115
 116        debug("%s(dev=%p)\n", __func__, dev);
 117
 118        ret = device_bind_driver_to_node(dev, "tegra186_clk", "tegra186_clk",
 119                                         dev_ofnode(dev), &child);
 120        if (ret)
 121                return ret;
 122
 123        ret = device_bind_driver_to_node(dev, "tegra186_reset",
 124                                         "tegra186_reset", dev_ofnode(dev),
 125                                         &child);
 126        if (ret)
 127                return ret;
 128
 129        ret = device_bind_driver_to_node(dev, "tegra186_power_domain",
 130                                         "tegra186_power_domain",
 131                                         dev_ofnode(dev), &child);
 132        if (ret)
 133                return ret;
 134
 135        ret = dm_scan_fdt_dev(dev);
 136        if (ret)
 137                return ret;
 138
 139        return 0;
 140}
 141
 142static ulong tegra186_bpmp_get_shmem(struct udevice *dev, int index)
 143{
 144        int ret;
 145        struct fdtdec_phandle_args args;
 146        fdt_addr_t reg;
 147
 148        ret = fdtdec_parse_phandle_with_args(gd->fdt_blob, dev_of_offset(dev),
 149                                              "shmem", NULL, 0, index, &args);
 150        if (ret < 0) {
 151                pr_err("fdtdec_parse_phandle_with_args() failed: %d\n", ret);
 152                return ret;
 153        }
 154
 155        reg = fdtdec_get_addr_size_auto_noparent(gd->fdt_blob, args.node,
 156                                                 "reg", 0, NULL, true);
 157        if (reg == FDT_ADDR_T_NONE) {
 158                pr_err("fdtdec_get_addr_size_auto_noparent() failed\n");
 159                return -ENODEV;
 160        }
 161
 162        return reg;
 163}
 164
 165static void tegra186_bpmp_ivc_notify(struct tegra_ivc *ivc)
 166{
 167        struct tegra186_bpmp *priv =
 168                container_of(ivc, struct tegra186_bpmp, ivc);
 169        int ret;
 170
 171        ret = mbox_send(&priv->mbox, NULL);
 172        if (ret)
 173                pr_err("mbox_send() failed: %d\n", ret);
 174}
 175
 176static int tegra186_bpmp_probe(struct udevice *dev)
 177{
 178        struct tegra186_bpmp *priv = dev_get_priv(dev);
 179        int ret;
 180        ulong tx_base, rx_base, start_time;
 181
 182        debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv);
 183
 184        ret = mbox_get_by_index(dev, 0, &priv->mbox);
 185        if (ret) {
 186                pr_err("mbox_get_by_index() failed: %d\n", ret);
 187                return ret;
 188        }
 189
 190        tx_base = tegra186_bpmp_get_shmem(dev, 0);
 191        if (IS_ERR_VALUE(tx_base)) {
 192                pr_err("tegra186_bpmp_get_shmem failed for tx_base\n");
 193                return tx_base;
 194        }
 195        rx_base = tegra186_bpmp_get_shmem(dev, 1);
 196        if (IS_ERR_VALUE(rx_base)) {
 197                pr_err("tegra186_bpmp_get_shmem failed for rx_base\n");
 198                return rx_base;
 199        }
 200        debug("shmem: rx=%lx, tx=%lx\n", rx_base, tx_base);
 201
 202        ret = tegra_ivc_init(&priv->ivc, rx_base, tx_base, BPMP_IVC_FRAME_COUNT,
 203                             BPMP_IVC_FRAME_SIZE, tegra186_bpmp_ivc_notify);
 204        if (ret) {
 205                pr_err("tegra_ivc_init() failed: %d\n", ret);
 206                return ret;
 207        }
 208
 209        tegra_ivc_channel_reset(&priv->ivc);
 210        start_time = timer_get_us();
 211        for (;;) {
 212                ret = tegra_ivc_channel_notified(&priv->ivc);
 213                if (!ret)
 214                        break;
 215
 216                /* Timeout 100ms */
 217                if ((timer_get_us() - start_time) > 100 * 1000) {
 218                        pr_err("Initial IVC reset timed out (%d)\n", ret);
 219                        ret = -ETIMEDOUT;
 220                        goto err_free_mbox;
 221                }
 222        }
 223
 224        return 0;
 225
 226err_free_mbox:
 227        mbox_free(&priv->mbox);
 228
 229        return ret;
 230}
 231
 232static int tegra186_bpmp_remove(struct udevice *dev)
 233{
 234        struct tegra186_bpmp *priv = dev_get_priv(dev);
 235
 236        debug("%s(dev=%p) (priv=%p)\n", __func__, dev, priv);
 237
 238        mbox_free(&priv->mbox);
 239
 240        return 0;
 241}
 242
 243static struct misc_ops tegra186_bpmp_ops = {
 244        .call = tegra186_bpmp_call,
 245};
 246
 247static const struct udevice_id tegra186_bpmp_ids[] = {
 248        { .compatible = "nvidia,tegra186-bpmp" },
 249        { }
 250};
 251
 252U_BOOT_DRIVER(tegra186_bpmp) = {
 253        .name           = "tegra186_bpmp",
 254        .id             = UCLASS_MISC,
 255        .of_match       = tegra186_bpmp_ids,
 256        .bind           = tegra186_bpmp_bind,
 257        .probe          = tegra186_bpmp_probe,
 258        .remove         = tegra186_bpmp_remove,
 259        .ops            = &tegra186_bpmp_ops,
 260        .priv_auto_alloc_size = sizeof(struct tegra186_bpmp),
 261};
 262