mirror of
git://nv-tegra.nvidia.com/linux-nv-oot.git
synced 2025-12-22 09:11:26 +03:00
This patch enables Tegra machine driver used on Jetson platform. Tegra210 and later Jetson platforms use the same machine driver. Following is the summary from the patch: * Enable the driver build in defconfig * Remove "tegra-alt" ASoC utility file * Renmae "tegra-alt" parser source file "tegra_asoc_dt_parser.c" to "tegra_asoc_machine.c" and move it to "tegra" directory. * Similarly "tegra_asoc_machine_alt.h" is renamed to "tegra_asoc_machine.h" and moved to "tegra". * Updated "tegra" ASoC utility files for supporting Tegra210 and later. * Rename machine driver file. * The older parsing code is removed. Bug 2845498 Change-Id: I42d9d2da59febb99ce00a29ce65d7a16f96b1327 Signed-off-by: Sameer Pujar <spujar@nvidia.com>
322 lines
7.6 KiB
C
322 lines
7.6 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* tegra_asoc_utils.c - Harmony machine ASoC driver
|
|
*
|
|
* Author: Stephen Warren <swarren@nvidia.com>
|
|
* Copyright (C) 2010,2012 - NVIDIA, Inc.
|
|
*/
|
|
|
|
#include <linux/clk.h>
|
|
#include <linux/device.h>
|
|
#include <linux/err.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/of.h>
|
|
|
|
#include "tegra_asoc_utils.h"
|
|
|
|
/*
|
|
* this will be used for platforms from Tegra210 onwards.
|
|
* odd rates: sample rates multiple of 11.025kHz
|
|
* even_rates: sample rates multiple of 8kHz
|
|
*/
|
|
enum rate_type {
|
|
ODD_RATE,
|
|
EVEN_RATE,
|
|
NUM_RATE_TYPE,
|
|
};
|
|
unsigned int tegra210_pll_base_rate[NUM_RATE_TYPE] = {338688000, 368640000};
|
|
unsigned int tegra186_pll_base_rate[NUM_RATE_TYPE] = {270950400, 245760000};
|
|
unsigned int default_pll_out_rate[NUM_RATE_TYPE] = {45158400, 49152000};
|
|
|
|
int tegra_asoc_utils_set_rate(struct tegra_asoc_utils_data *data, int srate,
|
|
int mclk)
|
|
{
|
|
int new_baseclock;
|
|
bool clk_change;
|
|
int err;
|
|
|
|
switch (srate) {
|
|
case 11025:
|
|
case 22050:
|
|
case 44100:
|
|
case 88200:
|
|
if (data->soc == TEGRA_ASOC_UTILS_SOC_TEGRA20)
|
|
new_baseclock = 56448000;
|
|
else if (data->soc == TEGRA_ASOC_UTILS_SOC_TEGRA30)
|
|
new_baseclock = 564480000;
|
|
else
|
|
new_baseclock = 282240000;
|
|
break;
|
|
case 8000:
|
|
case 16000:
|
|
case 32000:
|
|
case 48000:
|
|
case 64000:
|
|
case 96000:
|
|
if (data->soc == TEGRA_ASOC_UTILS_SOC_TEGRA20)
|
|
new_baseclock = 73728000;
|
|
else if (data->soc == TEGRA_ASOC_UTILS_SOC_TEGRA30)
|
|
new_baseclock = 552960000;
|
|
else
|
|
new_baseclock = 368640000;
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
clk_change = ((new_baseclock != data->set_baseclock) ||
|
|
(mclk != data->set_mclk));
|
|
if (!clk_change)
|
|
return 0;
|
|
|
|
data->set_baseclock = 0;
|
|
data->set_mclk = 0;
|
|
|
|
clk_disable_unprepare(data->clk_aud_mclk);
|
|
clk_disable_unprepare(data->clk_pll_out);
|
|
clk_disable_unprepare(data->clk_pll);
|
|
|
|
err = clk_set_rate(data->clk_pll, new_baseclock);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't set base pll rate: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
err = clk_set_rate(data->clk_pll_out, mclk);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't set pll_out rate: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
/* Don't set cdev1/extern1 rate; it's locked to pll_out */
|
|
|
|
err = clk_prepare_enable(data->clk_pll);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't enable pll: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
err = clk_prepare_enable(data->clk_pll_out);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't enable pll_out: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
err = clk_prepare_enable(data->clk_aud_mclk);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't enable aud_mclk: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
data->set_baseclock = new_baseclock;
|
|
data->set_mclk = mclk;
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(tegra_asoc_utils_set_rate);
|
|
|
|
int tegra_asoc_utils_set_ac97_rate(struct tegra_asoc_utils_data *data)
|
|
{
|
|
const int pll_rate = 73728000;
|
|
const int ac97_rate = 24576000;
|
|
int err;
|
|
|
|
clk_disable_unprepare(data->clk_aud_mclk);
|
|
clk_disable_unprepare(data->clk_pll_out);
|
|
clk_disable_unprepare(data->clk_pll);
|
|
|
|
/*
|
|
* AC97 rate is fixed at 24.576MHz and is used for both the host
|
|
* controller and the external codec
|
|
*/
|
|
err = clk_set_rate(data->clk_pll, pll_rate);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't set pll_a rate: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
err = clk_set_rate(data->clk_pll_out, ac97_rate);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't set pll_a_out0 rate: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
/* Don't set cdev1/extern1 rate; it's locked to pll_a_out0 */
|
|
|
|
err = clk_prepare_enable(data->clk_pll);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't enable pll_a: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
err = clk_prepare_enable(data->clk_pll_out);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't enable pll_a_out0: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
err = clk_prepare_enable(data->clk_aud_mclk);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't enable cdev1: %d\n", err);
|
|
return err;
|
|
}
|
|
|
|
data->set_baseclock = pll_rate;
|
|
data->set_mclk = ac97_rate;
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(tegra_asoc_utils_set_ac97_rate);
|
|
|
|
int tegra_asoc_utils_set_tegra210_rate(struct tegra_asoc_utils_data *data,
|
|
unsigned int sample_rate)
|
|
{
|
|
unsigned int new_pll_base, pll_out, aud_mclk = 0;
|
|
int err;
|
|
|
|
switch (sample_rate) {
|
|
case 11025:
|
|
case 22050:
|
|
case 44100:
|
|
case 88200:
|
|
case 176400:
|
|
new_pll_base = data->pll_base_rate[ODD_RATE];
|
|
pll_out = default_pll_out_rate[ODD_RATE];
|
|
break;
|
|
case 8000:
|
|
case 16000:
|
|
case 32000:
|
|
case 48000:
|
|
case 96000:
|
|
case 192000:
|
|
new_pll_base = data->pll_base_rate[EVEN_RATE];
|
|
pll_out = default_pll_out_rate[EVEN_RATE];
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* reduce pll_out rate to support lower sampling rates */
|
|
if (sample_rate <= 11025)
|
|
pll_out = pll_out >> 1;
|
|
if (data->mclk_fs)
|
|
aud_mclk = sample_rate * data->mclk_fs;
|
|
|
|
if (data->set_baseclock != new_pll_base) {
|
|
err = clk_set_rate(data->clk_pll, new_pll_base);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't set clk_pll rate: %d\n",
|
|
err);
|
|
return err;
|
|
}
|
|
data->set_baseclock = new_pll_base;
|
|
}
|
|
|
|
if (data->set_pll_out != pll_out) {
|
|
err = clk_set_rate(data->clk_pll_out, pll_out);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't set clk_pll_out rate: %d\n",
|
|
err);
|
|
return err;
|
|
}
|
|
data->set_pll_out = pll_out;
|
|
}
|
|
|
|
if (data->set_mclk != aud_mclk) {
|
|
err = clk_set_rate(data->clk_aud_mclk, aud_mclk);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't set clk_cdev1 rate: %d\n",
|
|
err);
|
|
return err;
|
|
}
|
|
data->set_mclk = aud_mclk;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(tegra_asoc_utils_set_tegra210_rate);
|
|
|
|
int tegra_asoc_utils_clk_enable(struct tegra_asoc_utils_data *data)
|
|
{
|
|
int err;
|
|
|
|
err = clk_prepare_enable(data->clk_aud_mclk);
|
|
if (err) {
|
|
dev_err(data->dev, "Can't enable clock aud_mclk\n");
|
|
return err;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(tegra_asoc_utils_clk_enable);
|
|
|
|
void tegra_asoc_utils_clk_disable(struct tegra_asoc_utils_data *data)
|
|
{
|
|
clk_disable_unprepare(data->clk_aud_mclk);
|
|
}
|
|
EXPORT_SYMBOL_GPL(tegra_asoc_utils_clk_disable);
|
|
|
|
int tegra_asoc_utils_init(struct tegra_asoc_utils_data *data,
|
|
struct device *dev)
|
|
{
|
|
int ret;
|
|
|
|
data->dev = dev;
|
|
|
|
if (of_machine_is_compatible("nvidia,tegra20"))
|
|
data->soc = TEGRA_ASOC_UTILS_SOC_TEGRA20;
|
|
else if (of_machine_is_compatible("nvidia,tegra30"))
|
|
data->soc = TEGRA_ASOC_UTILS_SOC_TEGRA30;
|
|
else if (of_machine_is_compatible("nvidia,tegra114"))
|
|
data->soc = TEGRA_ASOC_UTILS_SOC_TEGRA114;
|
|
else if (of_machine_is_compatible("nvidia,tegra124"))
|
|
data->soc = TEGRA_ASOC_UTILS_SOC_TEGRA124;
|
|
else if (of_machine_is_compatible("nvidia,tegra210"))
|
|
data->soc = TEGRA_ASOC_UTILS_SOC_TEGRA210;
|
|
else if (of_machine_is_compatible("nvidia,tegra186"))
|
|
data->soc = TEGRA_ASOC_UTILS_SOC_TEGRA186;
|
|
else if (of_machine_is_compatible("nvidia,tegra194"))
|
|
data->soc = TEGRA_ASOC_UTILS_SOC_TEGRA194;
|
|
else {
|
|
dev_err(data->dev, "SoC unknown to Tegra ASoC utils\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
data->clk_pll = devm_clk_get(dev, "pll_a");
|
|
if (IS_ERR(data->clk_pll)) {
|
|
dev_err(data->dev, "Can't retrieve clk pll_a\n");
|
|
return PTR_ERR(data->clk_pll);
|
|
}
|
|
|
|
data->clk_pll_out = devm_clk_get(dev, "pll_a_out0");
|
|
if (IS_ERR(data->clk_pll_out)) {
|
|
dev_err(data->dev, "Can't retrieve clk pll_a_out0\n");
|
|
return PTR_ERR(data->clk_pll_out);
|
|
}
|
|
|
|
data->clk_aud_mclk = devm_clk_get(dev, "extern1");
|
|
if (IS_ERR(data->clk_aud_mclk)) {
|
|
dev_err(data->dev, "Can't retrieve clk aud_mclk\n");
|
|
return PTR_ERR(data->clk_aud_mclk);
|
|
}
|
|
|
|
if (data->soc < TEGRA_ASOC_UTILS_SOC_TEGRA210) {
|
|
ret = tegra_asoc_utils_set_rate(data, 44100, 256 * 44100);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
if (data->soc < TEGRA_ASOC_UTILS_SOC_TEGRA186)
|
|
data->pll_base_rate = tegra210_pll_base_rate;
|
|
else
|
|
data->pll_base_rate = tegra186_pll_base_rate;
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(tegra_asoc_utils_init);
|
|
|
|
MODULE_AUTHOR("Stephen Warren <swarren@nvidia.com>");
|
|
MODULE_DESCRIPTION("Tegra ASoC utility code");
|
|
MODULE_LICENSE("GPL");
|