mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-11-03 21:48:15 +00:00 
			
		
		
		
	As part of bringing the master branch back in to next, we need to allow for all of these changes to exist here. Reported-by: Jonas Karlman <jonas@kwiboo.se> Signed-off-by: Tom Rini <trini@konsulko.com>
		
			
				
	
	
		
			194 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			194 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0
 | 
						|
/*
 | 
						|
 * Copyright (c) 2016, NVIDIA CORPORATION.
 | 
						|
 */
 | 
						|
 | 
						|
#include <log.h>
 | 
						|
#include <malloc.h>
 | 
						|
#include <asm/io.h>
 | 
						|
#include <dm.h>
 | 
						|
#include <mailbox-uclass.h>
 | 
						|
#include <dt-bindings/mailbox/tegra186-hsp.h>
 | 
						|
#include <linux/bitops.h>
 | 
						|
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING		0x380
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NSI_SHIFT	16
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NSI_MASK	0xf
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NDB_SHIFT	12
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NDB_MASK	0xf
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NAS_SHIFT	8
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NAS_MASK	0xf
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NSS_SHIFT	4
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NSS_MASK	0xf
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NSM_SHIFT	0
 | 
						|
#define TEGRA_HSP_INT_DIMENSIONING_NSM_MASK	0xf
 | 
						|
 | 
						|
#define TEGRA_HSP_DB_REG_TRIGGER	0x0
 | 
						|
#define TEGRA_HSP_DB_REG_ENABLE		0x4
 | 
						|
#define TEGRA_HSP_DB_REG_RAW		0x8
 | 
						|
#define TEGRA_HSP_DB_REG_PENDING	0xc
 | 
						|
 | 
						|
#define TEGRA_HSP_DB_ID_CCPLEX		1
 | 
						|
#define TEGRA_HSP_DB_ID_BPMP		3
 | 
						|
#define TEGRA_HSP_DB_ID_NUM		7
 | 
						|
 | 
						|
struct tegra_hsp {
 | 
						|
	fdt_addr_t regs;
 | 
						|
	uint32_t db_base;
 | 
						|
};
 | 
						|
 | 
						|
static uint32_t *tegra_hsp_reg(struct tegra_hsp *thsp, uint32_t db_id,
 | 
						|
			       uint32_t reg)
 | 
						|
{
 | 
						|
	return (uint32_t *)(thsp->regs + thsp->db_base + (db_id * 0x100) + reg);
 | 
						|
}
 | 
						|
 | 
						|
static uint32_t tegra_hsp_readl(struct tegra_hsp *thsp, uint32_t db_id,
 | 
						|
				uint32_t reg)
 | 
						|
{
 | 
						|
	uint32_t *r = tegra_hsp_reg(thsp, db_id, reg);
 | 
						|
	return readl(r);
 | 
						|
}
 | 
						|
 | 
						|
static void tegra_hsp_writel(struct tegra_hsp *thsp, uint32_t val,
 | 
						|
			     uint32_t db_id, uint32_t reg)
 | 
						|
{
 | 
						|
	uint32_t *r = tegra_hsp_reg(thsp, db_id, reg);
 | 
						|
 | 
						|
	writel(val, r);
 | 
						|
	readl(r);
 | 
						|
}
 | 
						|
 | 
						|
static int tegra_hsp_db_id(ulong chan_id)
 | 
						|
{
 | 
						|
	switch (chan_id) {
 | 
						|
	case (HSP_MBOX_TYPE_DB << 16) | HSP_DB_MASTER_BPMP:
 | 
						|
		return TEGRA_HSP_DB_ID_BPMP;
 | 
						|
	default:
 | 
						|
		debug("Invalid channel ID\n");
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static int tegra_hsp_of_xlate(struct mbox_chan *chan,
 | 
						|
			      struct ofnode_phandle_args *args)
 | 
						|
{
 | 
						|
	debug("%s(chan=%p)\n", __func__, chan);
 | 
						|
 | 
						|
	if (args->args_count != 2) {
 | 
						|
		debug("Invalid args_count: %d\n", args->args_count);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	chan->id = (args->args[0] << 16) | args->args[1];
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int tegra_hsp_request(struct mbox_chan *chan)
 | 
						|
{
 | 
						|
	int db_id;
 | 
						|
 | 
						|
	debug("%s(chan=%p)\n", __func__, chan);
 | 
						|
 | 
						|
	db_id = tegra_hsp_db_id(chan->id);
 | 
						|
	if (db_id < 0) {
 | 
						|
		debug("tegra_hsp_db_id() failed: %d\n", db_id);
 | 
						|
		return -EINVAL;
 | 
						|
	}
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int tegra_hsp_free(struct mbox_chan *chan)
 | 
						|
{
 | 
						|
	debug("%s(chan=%p)\n", __func__, chan);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int tegra_hsp_send(struct mbox_chan *chan, const void *data)
 | 
						|
{
 | 
						|
	struct tegra_hsp *thsp = dev_get_priv(chan->dev);
 | 
						|
	int db_id;
 | 
						|
 | 
						|
	debug("%s(chan=%p, data=%p)\n", __func__, chan, data);
 | 
						|
 | 
						|
	db_id = tegra_hsp_db_id(chan->id);
 | 
						|
	tegra_hsp_writel(thsp, 1, db_id, TEGRA_HSP_DB_REG_TRIGGER);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int tegra_hsp_recv(struct mbox_chan *chan, void *data)
 | 
						|
{
 | 
						|
	struct tegra_hsp *thsp = dev_get_priv(chan->dev);
 | 
						|
	uint32_t db_id = TEGRA_HSP_DB_ID_CCPLEX;
 | 
						|
	uint32_t val;
 | 
						|
 | 
						|
	debug("%s(chan=%p, data=%p)\n", __func__, chan, data);
 | 
						|
 | 
						|
	val = tegra_hsp_readl(thsp, db_id, TEGRA_HSP_DB_REG_RAW);
 | 
						|
	if (!(val & BIT(chan->id)))
 | 
						|
		return -ENODATA;
 | 
						|
 | 
						|
	tegra_hsp_writel(thsp, BIT(chan->id), db_id, TEGRA_HSP_DB_REG_RAW);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int tegra_hsp_bind(struct udevice *dev)
 | 
						|
{
 | 
						|
	debug("%s(dev=%p)\n", __func__, dev);
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static int tegra_hsp_probe(struct udevice *dev)
 | 
						|
{
 | 
						|
	struct tegra_hsp *thsp = dev_get_priv(dev);
 | 
						|
	u32 val;
 | 
						|
	int nr_sm, nr_ss, nr_as;
 | 
						|
 | 
						|
	debug("%s(dev=%p)\n", __func__, dev);
 | 
						|
 | 
						|
	thsp->regs = dev_read_addr(dev);
 | 
						|
	if (thsp->regs == FDT_ADDR_T_NONE)
 | 
						|
		return -ENODEV;
 | 
						|
 | 
						|
	val = readl(thsp->regs + TEGRA_HSP_INT_DIMENSIONING);
 | 
						|
	nr_sm = (val >> TEGRA_HSP_INT_DIMENSIONING_NSM_SHIFT) &
 | 
						|
		TEGRA_HSP_INT_DIMENSIONING_NSM_MASK;
 | 
						|
	nr_ss = (val >> TEGRA_HSP_INT_DIMENSIONING_NSS_SHIFT) &
 | 
						|
		TEGRA_HSP_INT_DIMENSIONING_NSS_MASK;
 | 
						|
	nr_as = (val >> TEGRA_HSP_INT_DIMENSIONING_NAS_SHIFT) &
 | 
						|
		TEGRA_HSP_INT_DIMENSIONING_NAS_MASK;
 | 
						|
 | 
						|
	thsp->db_base = (1 + (nr_sm >> 1) + nr_ss + nr_as) << 16;
 | 
						|
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static const struct udevice_id tegra_hsp_ids[] = {
 | 
						|
	{ .compatible = "nvidia,tegra186-hsp" },
 | 
						|
	{ }
 | 
						|
};
 | 
						|
 | 
						|
struct mbox_ops tegra_hsp_mbox_ops = {
 | 
						|
	.of_xlate = tegra_hsp_of_xlate,
 | 
						|
	.request = tegra_hsp_request,
 | 
						|
	.rfree = tegra_hsp_free,
 | 
						|
	.send = tegra_hsp_send,
 | 
						|
	.recv = tegra_hsp_recv,
 | 
						|
};
 | 
						|
 | 
						|
U_BOOT_DRIVER(tegra_hsp) = {
 | 
						|
	.name = "tegra-hsp",
 | 
						|
	.id = UCLASS_MAILBOX,
 | 
						|
	.of_match = tegra_hsp_ids,
 | 
						|
	.bind = tegra_hsp_bind,
 | 
						|
	.probe = tegra_hsp_probe,
 | 
						|
	.priv_auto	= sizeof(struct tegra_hsp),
 | 
						|
	.ops = &tegra_hsp_mbox_ops,
 | 
						|
};
 |