162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0+
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * This driver provides regmap to access to analog part of audio codec
462306a36Sopenharmony_ci * found on Allwinner A23, A31s, A33, H3 and A64 Socs
562306a36Sopenharmony_ci *
662306a36Sopenharmony_ci * Copyright 2016 Chen-Yu Tsai <wens@csie.org>
762306a36Sopenharmony_ci * Copyright (C) 2018 Vasily Khoruzhick <anarsoul@gmail.com>
862306a36Sopenharmony_ci */
962306a36Sopenharmony_ci
1062306a36Sopenharmony_ci#include <linux/io.h>
1162306a36Sopenharmony_ci#include <linux/kernel.h>
1262306a36Sopenharmony_ci#include <linux/module.h>
1362306a36Sopenharmony_ci#include <linux/regmap.h>
1462306a36Sopenharmony_ci
1562306a36Sopenharmony_ci#include "sun8i-adda-pr-regmap.h"
1662306a36Sopenharmony_ci
1762306a36Sopenharmony_ci/* Analog control register access bits */
1862306a36Sopenharmony_ci#define ADDA_PR			0x0		/* PRCM base + 0x1c0 */
1962306a36Sopenharmony_ci#define ADDA_PR_RESET			BIT(28)
2062306a36Sopenharmony_ci#define ADDA_PR_WRITE			BIT(24)
2162306a36Sopenharmony_ci#define ADDA_PR_ADDR_SHIFT		16
2262306a36Sopenharmony_ci#define ADDA_PR_ADDR_MASK		GENMASK(4, 0)
2362306a36Sopenharmony_ci#define ADDA_PR_DATA_IN_SHIFT		8
2462306a36Sopenharmony_ci#define ADDA_PR_DATA_IN_MASK		GENMASK(7, 0)
2562306a36Sopenharmony_ci#define ADDA_PR_DATA_OUT_SHIFT		0
2662306a36Sopenharmony_ci#define ADDA_PR_DATA_OUT_MASK		GENMASK(7, 0)
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci/* regmap access bits */
2962306a36Sopenharmony_cistatic int adda_reg_read(void *context, unsigned int reg, unsigned int *val)
3062306a36Sopenharmony_ci{
3162306a36Sopenharmony_ci	void __iomem *base = (void __iomem *)context;
3262306a36Sopenharmony_ci	u32 tmp;
3362306a36Sopenharmony_ci
3462306a36Sopenharmony_ci	/* De-assert reset */
3562306a36Sopenharmony_ci	writel(readl(base) | ADDA_PR_RESET, base);
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci	/* Clear write bit */
3862306a36Sopenharmony_ci	writel(readl(base) & ~ADDA_PR_WRITE, base);
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_ci	/* Set register address */
4162306a36Sopenharmony_ci	tmp = readl(base);
4262306a36Sopenharmony_ci	tmp &= ~(ADDA_PR_ADDR_MASK << ADDA_PR_ADDR_SHIFT);
4362306a36Sopenharmony_ci	tmp |= (reg & ADDA_PR_ADDR_MASK) << ADDA_PR_ADDR_SHIFT;
4462306a36Sopenharmony_ci	writel(tmp, base);
4562306a36Sopenharmony_ci
4662306a36Sopenharmony_ci	/* Read back value */
4762306a36Sopenharmony_ci	*val = readl(base) & ADDA_PR_DATA_OUT_MASK;
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci	return 0;
5062306a36Sopenharmony_ci}
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_cistatic int adda_reg_write(void *context, unsigned int reg, unsigned int val)
5362306a36Sopenharmony_ci{
5462306a36Sopenharmony_ci	void __iomem *base = (void __iomem *)context;
5562306a36Sopenharmony_ci	u32 tmp;
5662306a36Sopenharmony_ci
5762306a36Sopenharmony_ci	/* De-assert reset */
5862306a36Sopenharmony_ci	writel(readl(base) | ADDA_PR_RESET, base);
5962306a36Sopenharmony_ci
6062306a36Sopenharmony_ci	/* Set register address */
6162306a36Sopenharmony_ci	tmp = readl(base);
6262306a36Sopenharmony_ci	tmp &= ~(ADDA_PR_ADDR_MASK << ADDA_PR_ADDR_SHIFT);
6362306a36Sopenharmony_ci	tmp |= (reg & ADDA_PR_ADDR_MASK) << ADDA_PR_ADDR_SHIFT;
6462306a36Sopenharmony_ci	writel(tmp, base);
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_ci	/* Set data to write */
6762306a36Sopenharmony_ci	tmp = readl(base);
6862306a36Sopenharmony_ci	tmp &= ~(ADDA_PR_DATA_IN_MASK << ADDA_PR_DATA_IN_SHIFT);
6962306a36Sopenharmony_ci	tmp |= (val & ADDA_PR_DATA_IN_MASK) << ADDA_PR_DATA_IN_SHIFT;
7062306a36Sopenharmony_ci	writel(tmp, base);
7162306a36Sopenharmony_ci
7262306a36Sopenharmony_ci	/* Set write bit to signal a write */
7362306a36Sopenharmony_ci	writel(readl(base) | ADDA_PR_WRITE, base);
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci	/* Clear write bit */
7662306a36Sopenharmony_ci	writel(readl(base) & ~ADDA_PR_WRITE, base);
7762306a36Sopenharmony_ci
7862306a36Sopenharmony_ci	return 0;
7962306a36Sopenharmony_ci}
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_cistatic const struct regmap_config adda_pr_regmap_cfg = {
8262306a36Sopenharmony_ci	.name		= "adda-pr",
8362306a36Sopenharmony_ci	.reg_bits	= 5,
8462306a36Sopenharmony_ci	.reg_stride	= 1,
8562306a36Sopenharmony_ci	.val_bits	= 8,
8662306a36Sopenharmony_ci	.reg_read	= adda_reg_read,
8762306a36Sopenharmony_ci	.reg_write	= adda_reg_write,
8862306a36Sopenharmony_ci	.fast_io	= true,
8962306a36Sopenharmony_ci	.max_register	= 31,
9062306a36Sopenharmony_ci};
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_cistruct regmap *sun8i_adda_pr_regmap_init(struct device *dev,
9362306a36Sopenharmony_ci					 void __iomem *base)
9462306a36Sopenharmony_ci{
9562306a36Sopenharmony_ci	return devm_regmap_init(dev, NULL, base, &adda_pr_regmap_cfg);
9662306a36Sopenharmony_ci}
9762306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(sun8i_adda_pr_regmap_init);
9862306a36Sopenharmony_ci
9962306a36Sopenharmony_ciMODULE_DESCRIPTION("Allwinner analog audio codec regmap driver");
10062306a36Sopenharmony_ciMODULE_AUTHOR("Vasily Khoruzhick <anarsoul@gmail.com>");
10162306a36Sopenharmony_ciMODULE_LICENSE("GPL");
10262306a36Sopenharmony_ciMODULE_ALIAS("platform:sunxi-adda-pr");
103