2018-04-30 22:15:48 -07:00
|
|
|
/*
|
2018-08-05 04:40:32 -07:00
|
|
|
* Copyright (c) 2018 naehrwert
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-08-13 01:58:24 -07:00
|
|
|
#include "../soc/clock.h"
|
|
|
|
#include "../soc/t210.h"
|
|
|
|
#include "../utils/util.h"
|
|
|
|
#include "../storage/sdmmc.h"
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-11-10 04:11:42 -08:00
|
|
|
/* clock_t: reset, enable, source, index, clk_src, clk_div */
|
|
|
|
|
2018-08-05 04:40:32 -07:00
|
|
|
static const clock_t _clock_uart[] = {
|
2019-09-09 06:56:37 -07:00
|
|
|
/* UART A */ { CLK_RST_CONTROLLER_RST_DEVICES_L, CLK_RST_CONTROLLER_CLK_OUT_ENB_L, CLK_RST_CONTROLLER_CLK_SOURCE_UARTA, 6, 0, 2 },
|
|
|
|
/* UART B */ { CLK_RST_CONTROLLER_RST_DEVICES_L, CLK_RST_CONTROLLER_CLK_OUT_ENB_L, CLK_RST_CONTROLLER_CLK_SOURCE_UARTB, 7, 0, 2 },
|
|
|
|
/* UART C */ { CLK_RST_CONTROLLER_RST_DEVICES_H, CLK_RST_CONTROLLER_CLK_OUT_ENB_H, CLK_RST_CONTROLLER_CLK_SOURCE_UARTC, 23, 0, 2 },
|
|
|
|
/* UART D */ { CLK_RST_CONTROLLER_RST_DEVICES_U, CLK_RST_CONTROLLER_CLK_OUT_ENB_U, CLK_RST_CONTROLLER_CLK_SOURCE_UARTD, 1, 0, 2 },
|
|
|
|
/* UART E */ { CLK_RST_CONTROLLER_RST_DEVICES_Y, CLK_RST_CONTROLLER_CLK_OUT_ENB_Y, CLK_RST_CONTROLLER_CLK_SOURCE_UARTAPE, 20, 0, 2 }
|
2018-04-30 22:15:48 -07:00
|
|
|
};
|
|
|
|
|
2019-12-04 11:31:39 -08:00
|
|
|
//I2C default parameters - TLOW: 4, THIGH: 2, DEBOUNCE: 0 FM_DIV: 26.
|
2018-04-30 22:15:48 -07:00
|
|
|
static const clock_t _clock_i2c[] = {
|
2019-12-04 11:32:51 -08:00
|
|
|
/* I2C1 */ { CLK_RST_CONTROLLER_RST_DEVICES_L, CLK_RST_CONTROLLER_CLK_OUT_ENB_L, CLK_RST_CONTROLLER_CLK_SOURCE_I2C1, 12, 0, 19 }, //20.4MHz -> 100KHz
|
|
|
|
/* I2C2 */ { CLK_RST_CONTROLLER_RST_DEVICES_H, CLK_RST_CONTROLLER_CLK_OUT_ENB_H, CLK_RST_CONTROLLER_CLK_SOURCE_I2C2, 22, 0, 4 }, //81.6MHz -> 400KHz
|
|
|
|
/* I2C3 */ { CLK_RST_CONTROLLER_RST_DEVICES_U, CLK_RST_CONTROLLER_CLK_OUT_ENB_U, CLK_RST_CONTROLLER_CLK_SOURCE_I2C3, 3, 0, 4 }, //81.6MHz -> 400KHz
|
|
|
|
/* I2C4 */ { CLK_RST_CONTROLLER_RST_DEVICES_V, CLK_RST_CONTROLLER_CLK_OUT_ENB_V, CLK_RST_CONTROLLER_CLK_SOURCE_I2C4, 7, 0, 19 }, //20.4MHz -> 100KHz
|
|
|
|
/* I2C5 */ { CLK_RST_CONTROLLER_RST_DEVICES_H, CLK_RST_CONTROLLER_CLK_OUT_ENB_H, CLK_RST_CONTROLLER_CLK_SOURCE_I2C5, 15, 0, 4 }, //81.6MHz -> 400KHz
|
|
|
|
/* I2C6 */ { CLK_RST_CONTROLLER_RST_DEVICES_X, CLK_RST_CONTROLLER_CLK_OUT_ENB_X, CLK_RST_CONTROLLER_CLK_SOURCE_I2C6, 6, 0, 19 } //20.4MHz -> 100KHz
|
2018-04-30 22:15:48 -07:00
|
|
|
};
|
|
|
|
|
2018-11-10 04:11:42 -08:00
|
|
|
static clock_t _clock_se = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_V, CLK_RST_CONTROLLER_CLK_OUT_ENB_V, CLK_RST_CONTROLLER_CLK_SOURCE_SE, 31, 0, 0
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
2019-06-29 17:15:46 -07:00
|
|
|
|
|
|
|
static clock_t _clock_tzram = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_V, CLK_RST_CONTROLLER_CLK_OUT_ENB_V, CLK_NO_SOURCE, 30, 0, 0
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-11-10 04:11:42 -08:00
|
|
|
static clock_t _clock_host1x = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_L, CLK_RST_CONTROLLER_CLK_OUT_ENB_L, CLK_RST_CONTROLLER_CLK_SOURCE_HOST1X, 28, 4, 3
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
|
|
|
static clock_t _clock_tsec = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_U, CLK_RST_CONTROLLER_CLK_OUT_ENB_U, CLK_RST_CONTROLLER_CLK_SOURCE_TSEC, 19, 0, 2
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
|
|
|
static clock_t _clock_sor_safe = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_Y, CLK_RST_CONTROLLER_CLK_OUT_ENB_Y, CLK_NO_SOURCE, 30, 0, 0
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
|
|
|
static clock_t _clock_sor0 = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_X, CLK_RST_CONTROLLER_CLK_OUT_ENB_X, CLK_NO_SOURCE, 22, 0, 0
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
|
|
|
static clock_t _clock_sor1 = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_X, CLK_RST_CONTROLLER_CLK_OUT_ENB_X, CLK_RST_CONTROLLER_CLK_SOURCE_SOR1, 23, 0, 2
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
|
|
|
static clock_t _clock_kfuse = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_H, CLK_RST_CONTROLLER_CLK_OUT_ENB_H, CLK_NO_SOURCE, 8, 0, 0
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-11-10 04:11:42 -08:00
|
|
|
static clock_t _clock_cl_dvfs = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_W, CLK_RST_CONTROLLER_CLK_OUT_ENB_W, CLK_NO_SOURCE, 27, 0, 0
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
|
|
|
static clock_t _clock_coresight = {
|
2019-09-09 06:56:37 -07:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_U, CLK_RST_CONTROLLER_CLK_OUT_ENB_U, CLK_RST_CONTROLLER_CLK_SOURCE_CSITE, 9, 0, 4
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-11-10 04:11:42 -08:00
|
|
|
static clock_t _clock_pwm = {
|
2019-12-04 11:31:39 -08:00
|
|
|
CLK_RST_CONTROLLER_RST_DEVICES_L, CLK_RST_CONTROLLER_CLK_OUT_ENB_L, CLK_RST_CONTROLLER_CLK_SOURCE_PWM, 17, 6, 4 // Fref: 6.2MHz.
|
2018-11-10 04:11:42 -08:00
|
|
|
};
|
2018-09-18 14:01:42 -07:00
|
|
|
|
2018-04-30 22:15:48 -07:00
|
|
|
void clock_enable(const clock_t *clk)
|
|
|
|
{
|
2018-08-05 04:40:32 -07:00
|
|
|
// Put clock into reset.
|
2018-06-08 02:42:24 -07:00
|
|
|
CLOCK(clk->reset) = (CLOCK(clk->reset) & ~(1 << clk->index)) | (1 << clk->index);
|
2018-08-05 04:40:32 -07:00
|
|
|
// Disable.
|
2018-04-30 22:15:48 -07:00
|
|
|
CLOCK(clk->enable) &= ~(1 << clk->index);
|
2018-08-05 04:40:32 -07:00
|
|
|
// Configure clock source if required.
|
2018-04-30 22:15:48 -07:00
|
|
|
if (clk->source)
|
|
|
|
CLOCK(clk->source) = clk->clk_div | (clk->clk_src << 29);
|
2018-08-05 04:40:32 -07:00
|
|
|
// Enable.
|
2018-06-08 02:42:24 -07:00
|
|
|
CLOCK(clk->enable) = (CLOCK(clk->enable) & ~(1 << clk->index)) | (1 << clk->index);
|
2019-12-04 11:32:51 -08:00
|
|
|
usleep(2);
|
|
|
|
|
2018-08-05 04:40:32 -07:00
|
|
|
// Take clock off reset.
|
2018-04-30 22:15:48 -07:00
|
|
|
CLOCK(clk->reset) &= ~(1 << clk->index);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_disable(const clock_t *clk)
|
|
|
|
{
|
2018-08-05 04:40:32 -07:00
|
|
|
// Put clock into reset.
|
2018-06-08 02:42:24 -07:00
|
|
|
CLOCK(clk->reset) = (CLOCK(clk->reset) & ~(1 << clk->index)) | (1 << clk->index);
|
2018-08-05 04:40:32 -07:00
|
|
|
// Disable.
|
2018-04-30 22:15:48 -07:00
|
|
|
CLOCK(clk->enable) &= ~(1 << clk->index);
|
|
|
|
}
|
|
|
|
|
2018-09-18 13:38:54 -07:00
|
|
|
void clock_enable_fuse(bool enable)
|
2018-04-30 22:15:48 -07:00
|
|
|
{
|
2018-06-08 02:42:24 -07:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_MISC_CLK_ENB) = (CLOCK(CLK_RST_CONTROLLER_MISC_CLK_ENB) & 0xEFFFFFFF) | ((enable & 1) << 28);
|
2018-04-30 22:15:48 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void clock_enable_uart(u32 idx)
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_uart[idx]);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_enable_i2c(u32 idx)
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_i2c[idx]);
|
|
|
|
}
|
|
|
|
|
2018-09-18 13:38:54 -07:00
|
|
|
void clock_disable_i2c(u32 idx)
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_i2c[idx]);
|
|
|
|
}
|
|
|
|
|
2018-04-30 22:15:48 -07:00
|
|
|
void clock_enable_se()
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_se);
|
|
|
|
}
|
|
|
|
|
2019-06-29 17:15:46 -07:00
|
|
|
void clock_enable_tzram()
|
2018-09-18 13:38:54 -07:00
|
|
|
{
|
2019-06-29 17:15:46 -07:00
|
|
|
clock_enable(&_clock_tzram);
|
2018-09-18 13:38:54 -07:00
|
|
|
}
|
|
|
|
|
2018-04-30 22:15:48 -07:00
|
|
|
void clock_enable_host1x()
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_host1x);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_disable_host1x()
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_host1x);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_enable_tsec()
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_tsec);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_disable_tsec()
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_tsec);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_enable_sor_safe()
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_sor_safe);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_disable_sor_safe()
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_sor_safe);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_enable_sor0()
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_sor0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_disable_sor0()
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_sor0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_enable_sor1()
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_sor1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_disable_sor1()
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_sor1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_enable_kfuse()
|
|
|
|
{
|
|
|
|
//clock_enable(&_clock_kfuse);
|
2018-11-10 04:11:42 -08:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEVICES_H) = (CLOCK(CLK_RST_CONTROLLER_RST_DEVICES_H) & 0xFFFFFEFF) | 0x100;
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_OUT_ENB_H) &= 0xFFFFFEFF;
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_OUT_ENB_H) = (CLOCK(CLK_RST_CONTROLLER_CLK_OUT_ENB_H) & 0xFFFFFEFF) | 0x100;
|
2018-07-04 08:39:26 -07:00
|
|
|
usleep(10);
|
2018-11-10 04:11:42 -08:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEVICES_H) &= 0xFFFFFEFF;
|
2018-07-04 08:39:26 -07:00
|
|
|
usleep(20);
|
2018-04-30 22:15:48 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void clock_disable_kfuse()
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_kfuse);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_enable_cl_dvfs()
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_cl_dvfs);
|
|
|
|
}
|
|
|
|
|
2018-08-05 04:40:32 -07:00
|
|
|
void clock_disable_cl_dvfs()
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_cl_dvfs);
|
|
|
|
}
|
|
|
|
|
2018-04-30 22:15:48 -07:00
|
|
|
void clock_enable_coresight()
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_coresight);
|
|
|
|
}
|
|
|
|
|
2018-09-18 13:38:54 -07:00
|
|
|
void clock_disable_coresight()
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_coresight);
|
|
|
|
}
|
|
|
|
|
2018-09-18 14:01:42 -07:00
|
|
|
void clock_enable_pwm()
|
|
|
|
{
|
|
|
|
clock_enable(&_clock_pwm);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_disable_pwm()
|
|
|
|
{
|
|
|
|
clock_disable(&_clock_pwm);
|
|
|
|
}
|
|
|
|
|
2020-01-06 20:46:22 -08:00
|
|
|
void clock_enable_pllc(u32 divn)
|
|
|
|
{
|
|
|
|
u8 pll_divn_curr = (CLOCK(CLK_RST_CONTROLLER_PLLC_BASE) >> 10) & 0xFF;
|
|
|
|
|
|
|
|
// Check if already enabled and configured.
|
|
|
|
if ((CLOCK(CLK_RST_CONTROLLER_PLLC_BASE) & PLLCX_BASE_ENABLE) && (pll_divn_curr == divn))
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Take PLLC out of reset and set basic misc parameters.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_MISC) =
|
|
|
|
((CLOCK(CLK_RST_CONTROLLER_PLLC_MISC) & 0xFFF0000F) & ~PLLC_MISC_RESET) | (0x80000 << 4); // PLLC_EXT_FRU.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_MISC_2) |= 0xF0 << 8; // PLLC_FLL_LD_MEM.
|
|
|
|
|
|
|
|
// Disable PLL and IDDQ in case they are on.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_BASE) &= ~PLLCX_BASE_ENABLE;
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_MISC_1) &= ~PLLC_MISC1_IDDQ;
|
|
|
|
usleep(10);
|
|
|
|
|
|
|
|
// Set PLLC4 dividers.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_BASE) = 4 | (divn << 10); // DIVM: 4, DIVP: 1.
|
|
|
|
|
|
|
|
// Enable PLLC4 and wait for Phase and Frequency lock.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_BASE) |= PLLCX_BASE_ENABLE;
|
|
|
|
while (!(CLOCK(CLK_RST_CONTROLLER_PLLC_BASE) & PLLCX_BASE_LOCK))
|
|
|
|
;
|
|
|
|
|
|
|
|
// Disable PLLC_OUT1, enable reset and set div to 1.5.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_OUT) = (1 << 8);
|
|
|
|
|
|
|
|
// Enable PLLC_OUT1 and bring it out of reset.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_OUT) |= (PLLC_OUT1_CLKEN | PLLC_OUT1_RSTN_CLR);
|
|
|
|
msleep(1); // Wait a bit for clock source change.
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_disable_pllc()
|
|
|
|
{
|
|
|
|
// Disable PLLC and PLLC_OUT1.
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_OUT) &= ~(PLLC_OUT1_CLKEN | PLLC_OUT1_RSTN_CLR);
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_BASE) &= ~PLLCX_BASE_ENABLE;
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_BASE) |= PLLCX_BASE_REF_DIS;
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_MISC_1) |= PLLC_MISC1_IDDQ;
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_PLLC_MISC) |= PLLC_MISC_RESET;
|
|
|
|
usleep(10);
|
|
|
|
}
|
|
|
|
|
2018-08-05 04:40:32 -07:00
|
|
|
#define L_SWR_SDMMC1_RST (1 << 14)
|
|
|
|
#define L_SWR_SDMMC2_RST (1 << 9)
|
|
|
|
#define L_SWR_SDMMC4_RST (1 << 15)
|
|
|
|
#define U_SWR_SDMMC3_RST (1 << 5)
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-08-05 04:40:32 -07:00
|
|
|
#define L_CLK_ENB_SDMMC1 (1 << 14)
|
|
|
|
#define L_CLK_ENB_SDMMC2 (1 << 9)
|
|
|
|
#define L_CLK_ENB_SDMMC4 (1 << 15)
|
|
|
|
#define U_CLK_ENB_SDMMC3 (1 << 5)
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-08-05 04:40:32 -07:00
|
|
|
#define L_SET_SDMMC1_RST (1 << 14)
|
|
|
|
#define L_SET_SDMMC2_RST (1 << 9)
|
|
|
|
#define L_SET_SDMMC4_RST (1 << 15)
|
|
|
|
#define U_SET_SDMMC3_RST (1 << 5)
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-08-05 04:40:32 -07:00
|
|
|
#define L_CLR_SDMMC1_RST (1 << 14)
|
|
|
|
#define L_CLR_SDMMC2_RST (1 << 9)
|
|
|
|
#define L_CLR_SDMMC4_RST (1 << 15)
|
|
|
|
#define U_CLR_SDMMC3_RST (1 << 5)
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-08-05 04:40:32 -07:00
|
|
|
#define L_SET_CLK_ENB_SDMMC1 (1 << 14)
|
|
|
|
#define L_SET_CLK_ENB_SDMMC2 (1 << 9)
|
|
|
|
#define L_SET_CLK_ENB_SDMMC4 (1 << 15)
|
|
|
|
#define U_SET_CLK_ENB_SDMMC3 (1 << 5)
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2018-08-05 04:40:32 -07:00
|
|
|
#define L_CLR_CLK_ENB_SDMMC1 (1 << 14)
|
|
|
|
#define L_CLR_CLK_ENB_SDMMC2 (1 << 9)
|
|
|
|
#define L_CLR_CLK_ENB_SDMMC4 (1 << 15)
|
|
|
|
#define U_CLR_CLK_ENB_SDMMC3 (1 << 5)
|
2018-04-30 22:15:48 -07:00
|
|
|
|
|
|
|
static int _clock_sdmmc_is_reset(u32 id)
|
|
|
|
{
|
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case SDMMC_1:
|
|
|
|
return CLOCK(CLK_RST_CONTROLLER_RST_DEVICES_L) & L_SWR_SDMMC1_RST;
|
|
|
|
case SDMMC_2:
|
|
|
|
return CLOCK(CLK_RST_CONTROLLER_RST_DEVICES_L) & L_SWR_SDMMC2_RST;
|
|
|
|
case SDMMC_3:
|
|
|
|
return CLOCK(CLK_RST_CONTROLLER_RST_DEVICES_U) & U_SWR_SDMMC3_RST;
|
|
|
|
case SDMMC_4:
|
|
|
|
return CLOCK(CLK_RST_CONTROLLER_RST_DEVICES_L) & L_SWR_SDMMC4_RST;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _clock_sdmmc_set_reset(u32 id)
|
|
|
|
{
|
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case SDMMC_1:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEV_L_SET) = L_SET_SDMMC1_RST;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_2:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEV_L_SET) = L_SET_SDMMC2_RST;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_3:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEV_U_SET) = U_SET_SDMMC3_RST;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_4:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEV_L_SET) = L_SET_SDMMC4_RST;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _clock_sdmmc_clear_reset(u32 id)
|
|
|
|
{
|
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case SDMMC_1:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEV_L_CLR) = L_CLR_SDMMC1_RST;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_2:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEV_L_CLR) = L_CLR_SDMMC2_RST;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_3:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEV_U_CLR) = U_CLR_SDMMC3_RST;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_4:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_RST_DEV_L_CLR) = L_CLR_SDMMC4_RST;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _clock_sdmmc_is_enabled(u32 id)
|
|
|
|
{
|
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case SDMMC_1:
|
|
|
|
return CLOCK(CLK_RST_CONTROLLER_CLK_OUT_ENB_L) & L_CLK_ENB_SDMMC1;
|
|
|
|
case SDMMC_2:
|
|
|
|
return CLOCK(CLK_RST_CONTROLLER_CLK_OUT_ENB_L) & L_CLK_ENB_SDMMC2;
|
|
|
|
case SDMMC_3:
|
|
|
|
return CLOCK(CLK_RST_CONTROLLER_CLK_OUT_ENB_U) & U_CLK_ENB_SDMMC3;
|
|
|
|
case SDMMC_4:
|
|
|
|
return CLOCK(CLK_RST_CONTROLLER_CLK_OUT_ENB_L) & L_CLK_ENB_SDMMC4;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-06-08 02:42:24 -07:00
|
|
|
static void _clock_sdmmc_set_enable(u32 id)
|
2018-04-30 22:15:48 -07:00
|
|
|
{
|
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case SDMMC_1:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_ENB_L_SET) = L_SET_CLK_ENB_SDMMC1;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_2:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_ENB_L_SET) = L_SET_CLK_ENB_SDMMC2;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_3:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_ENB_U_SET) = U_SET_CLK_ENB_SDMMC3;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_4:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_ENB_L_SET) = L_SET_CLK_ENB_SDMMC4;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-08 02:42:24 -07:00
|
|
|
static void _clock_sdmmc_clear_enable(u32 id)
|
2018-04-30 22:15:48 -07:00
|
|
|
{
|
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case SDMMC_1:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_ENB_L_CLR) = L_CLR_CLK_ENB_SDMMC1;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_2:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_ENB_L_CLR) = L_CLR_CLK_ENB_SDMMC2;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_3:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_ENB_U_CLR) = U_CLR_CLK_ENB_SDMMC3;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case SDMMC_4:
|
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_ENB_L_CLR) = L_CLR_CLK_ENB_SDMMC4;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static u32 _clock_sdmmc_table[8] = { 0 };
|
|
|
|
|
2019-04-22 17:34:39 -07:00
|
|
|
#define PLLP_OUT0 0x0
|
2019-12-04 11:31:39 -08:00
|
|
|
static int _clock_sdmmc_config_clock_host(u32 *pout, u32 id, u32 val)
|
2018-04-30 22:15:48 -07:00
|
|
|
{
|
|
|
|
u32 divisor = 0;
|
2019-04-22 17:34:39 -07:00
|
|
|
u32 source = PLLP_OUT0;
|
2018-04-30 22:15:48 -07:00
|
|
|
|
2019-09-09 06:56:37 -07:00
|
|
|
// Get IO clock divisor.
|
2018-04-30 22:15:48 -07:00
|
|
|
switch (val)
|
|
|
|
{
|
|
|
|
case 25000:
|
|
|
|
*pout = 24728;
|
2019-09-09 06:56:37 -07:00
|
|
|
divisor = 31; // 16.5 div.
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case 26000:
|
|
|
|
*pout = 25500;
|
2019-09-09 06:56:37 -07:00
|
|
|
divisor = 30; // 16 div.
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case 40800:
|
|
|
|
*pout = 40800;
|
2019-09-09 06:56:37 -07:00
|
|
|
divisor = 18; // 10 div.
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case 50000:
|
|
|
|
*pout = 48000;
|
2019-09-09 06:56:37 -07:00
|
|
|
divisor = 15; // 8.5 div.
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case 52000:
|
|
|
|
*pout = 51000;
|
2019-09-09 06:56:37 -07:00
|
|
|
divisor = 14; // 8 div.
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case 100000:
|
|
|
|
*pout = 90667;
|
2019-09-09 06:56:37 -07:00
|
|
|
divisor = 7; // 4.5 div.
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case 200000:
|
|
|
|
*pout = 163200;
|
2019-09-09 06:56:37 -07:00
|
|
|
divisor = 3; // 2.5 div.
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case 208000:
|
|
|
|
*pout = 204000;
|
2019-09-09 06:56:37 -07:00
|
|
|
divisor = 2; // 2 div.
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
default:
|
2018-08-22 18:37:02 -07:00
|
|
|
*pout = 24728;
|
2019-09-09 06:56:37 -07:00
|
|
|
divisor = 31; // 16.5 div.
|
2018-04-30 22:15:48 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
_clock_sdmmc_table[2 * id] = val;
|
|
|
|
_clock_sdmmc_table[2 * id + 1] = *pout;
|
|
|
|
|
2019-12-04 11:31:39 -08:00
|
|
|
// Set SDMMC clock.
|
2018-04-30 22:15:48 -07:00
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case SDMMC_1:
|
2019-04-22 17:34:39 -07:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_SOURCE_SDMMC1) = (source << 29) | divisor;
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case SDMMC_2:
|
2019-04-22 17:34:39 -07:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_SOURCE_SDMMC2) = (source << 29) | divisor;
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case SDMMC_3:
|
2019-04-22 17:34:39 -07:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_SOURCE_SDMMC3) = (source << 29) | divisor;
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
case SDMMC_4:
|
2019-04-22 17:34:39 -07:00
|
|
|
CLOCK(CLK_RST_CONTROLLER_CLK_SOURCE_SDMMC4) = (source << 29) | divisor;
|
2018-04-30 22:15:48 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_sdmmc_config_clock_source(u32 *pout, u32 id, u32 val)
|
|
|
|
{
|
|
|
|
if (_clock_sdmmc_table[2 * id] == val)
|
|
|
|
{
|
|
|
|
*pout = _clock_sdmmc_table[2 * id + 1];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int is_enabled = _clock_sdmmc_is_enabled(id);
|
|
|
|
if (is_enabled)
|
|
|
|
_clock_sdmmc_clear_enable(id);
|
2019-12-04 11:31:39 -08:00
|
|
|
_clock_sdmmc_config_clock_host(pout, id, val);
|
2018-04-30 22:15:48 -07:00
|
|
|
if (is_enabled)
|
|
|
|
_clock_sdmmc_set_enable(id);
|
|
|
|
_clock_sdmmc_is_reset(id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-04 11:31:39 -08:00
|
|
|
void clock_sdmmc_get_card_clock_div(u32 *pout, u16 *pdivisor, u32 type)
|
2018-04-30 22:15:48 -07:00
|
|
|
{
|
2019-12-04 11:31:39 -08:00
|
|
|
// Get Card clock divisor.
|
2018-04-30 22:15:48 -07:00
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
*pout = 26000;
|
|
|
|
*pdivisor = 66;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
*pout = 26000;
|
|
|
|
*pdivisor = 1;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
*pout = 52000;
|
|
|
|
*pdivisor = 1;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
case 4:
|
|
|
|
case 11:
|
2019-04-22 17:34:39 -07:00
|
|
|
*pout = 200000;
|
2018-04-30 22:15:48 -07:00
|
|
|
*pdivisor = 1;
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
*pout = 25000;
|
|
|
|
*pdivisor = 64;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case 6:
|
|
|
|
case 8:
|
|
|
|
*pout = 25000;
|
|
|
|
*pdivisor = 1;
|
|
|
|
break;
|
|
|
|
case 7:
|
|
|
|
*pout = 50000;
|
|
|
|
*pdivisor = 1;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case 10:
|
|
|
|
*pout = 100000;
|
|
|
|
*pdivisor = 1;
|
2019-04-13 16:19:04 -07:00
|
|
|
break;
|
2018-04-30 22:15:48 -07:00
|
|
|
case 13:
|
|
|
|
*pout = 40800;
|
|
|
|
*pdivisor = 1;
|
|
|
|
break;
|
|
|
|
case 14:
|
|
|
|
*pout = 200000;
|
|
|
|
*pdivisor = 2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int clock_sdmmc_is_not_reset_and_enabled(u32 id)
|
|
|
|
{
|
|
|
|
return !_clock_sdmmc_is_reset(id) && _clock_sdmmc_is_enabled(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_sdmmc_enable(u32 id, u32 val)
|
|
|
|
{
|
|
|
|
u32 div = 0;
|
|
|
|
|
|
|
|
if (_clock_sdmmc_is_enabled(id))
|
|
|
|
_clock_sdmmc_clear_enable(id);
|
|
|
|
_clock_sdmmc_set_reset(id);
|
2019-12-04 11:31:39 -08:00
|
|
|
_clock_sdmmc_config_clock_host(&div, id, val);
|
2018-04-30 22:15:48 -07:00
|
|
|
_clock_sdmmc_set_enable(id);
|
|
|
|
_clock_sdmmc_is_reset(id);
|
2018-07-04 08:39:26 -07:00
|
|
|
usleep((100000 + div - 1) / div);
|
2018-04-30 22:15:48 -07:00
|
|
|
_clock_sdmmc_clear_reset(id);
|
|
|
|
_clock_sdmmc_is_reset(id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clock_sdmmc_disable(u32 id)
|
|
|
|
{
|
|
|
|
_clock_sdmmc_set_reset(id);
|
|
|
|
_clock_sdmmc_clear_enable(id);
|
|
|
|
_clock_sdmmc_is_reset(id);
|
|
|
|
}
|