freebsd-dev/sys/arm/freescale/imx/imx6_ccm.c
Oleksandr Tymoshenko da21a623dd Add mode selection to iMX6 IPU driver
- Configure ipu1_di0 tob e sourced from the VIDEO_PLL(PLL5) and hardcode
  frequency to (455000000/3)Mhz. This value, further divided, can yield
  frequencies close enough to support 1080p, 720p, 1024x768, and 640x480
  modes. This is not ideal but it's an improvement comparing to the only
  hardcoded 1024x768 mode.

- Fix memory leaks if attach method failed
- Print EDID when -v passed to the kernel
2020-06-10 22:00:31 +00:00

522 lines
12 KiB
C

/*-
* SPDX-License-Identifier: BSD-2-Clause-FreeBSD
*
* Copyright (c) 2013 Ian Lepore <ian@freebsd.org>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
#include <sys/cdefs.h>
__FBSDID("$FreeBSD$");
/*
* Clocks and power control driver for Freescale i.MX6 family of SoCs.
*/
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/module.h>
#include <sys/bus.h>
#include <sys/rman.h>
#include <dev/ofw/ofw_bus.h>
#include <dev/ofw/ofw_bus_subr.h>
#include <machine/bus.h>
#include <arm/freescale/imx/imx6_anatopreg.h>
#include <arm/freescale/imx/imx6_anatopvar.h>
#include <arm/freescale/imx/imx6_ccmreg.h>
#include <arm/freescale/imx/imx_machdep.h>
#include <arm/freescale/imx/imx_ccmvar.h>
#ifndef CCGR_CLK_MODE_ALWAYS
#define CCGR_CLK_MODE_OFF 0
#define CCGR_CLK_MODE_RUNMODE 1
#define CCGR_CLK_MODE_ALWAYS 3
#endif
struct ccm_softc {
device_t dev;
struct resource *mem_res;
};
static struct ccm_softc *ccm_sc;
static inline uint32_t
RD4(struct ccm_softc *sc, bus_size_t off)
{
return (bus_read_4(sc->mem_res, off));
}
static inline void
WR4(struct ccm_softc *sc, bus_size_t off, uint32_t val)
{
bus_write_4(sc->mem_res, off, val);
}
/*
* Until we have a fully functional ccm driver which implements the fdt_clock
* interface, use the age-old workaround of unconditionally enabling the clocks
* for devices we might need to use. The SoC defaults to most clocks enabled,
* but the rom boot code and u-boot disable a few of them. We turn on only
* what's needed to run the chip plus devices we have drivers for, and turn off
* devices we don't yet have drivers for. (Note that USB is not turned on here
* because that is one we do when the driver asks for it.)
*/
static void
ccm_init_gates(struct ccm_softc *sc)
{
uint32_t reg;
/* ahpbdma, aipstz 1 & 2 buses */
reg = CCGR0_AIPS_TZ1 | CCGR0_AIPS_TZ2 | CCGR0_ABPHDMA;
WR4(sc, CCM_CCGR0, reg);
/* enet, epit, gpt, spi */
reg = CCGR1_ENET | CCGR1_EPIT1 | CCGR1_GPT | CCGR1_ECSPI1 |
CCGR1_ECSPI2 | CCGR1_ECSPI3 | CCGR1_ECSPI4 | CCGR1_ECSPI5;
WR4(sc, CCM_CCGR1, reg);
/* ipmux & ipsync (bridges), iomux, i2c */
reg = CCGR2_I2C1 | CCGR2_I2C2 | CCGR2_I2C3 | CCGR2_IIM |
CCGR2_IOMUX_IPT | CCGR2_IPMUX1 | CCGR2_IPMUX2 | CCGR2_IPMUX3 |
CCGR2_IPSYNC_IP2APB_TZASC1 | CCGR2_IPSYNC_IP2APB_TZASC2 |
CCGR2_IPSYNC_VDOA;
WR4(sc, CCM_CCGR2, reg);
/* DDR memory controller */
reg = CCGR3_OCRAM | CCGR3_MMDC_CORE_IPG |
CCGR3_MMDC_CORE_ACLK_FAST | CCGR3_CG11 | CCGR3_CG13;
WR4(sc, CCM_CCGR3, reg);
/* pl301 bus crossbar */
reg = CCGR4_PL301_MX6QFAST1_S133 |
CCGR4_PL301_MX6QPER1_BCH | CCGR4_PL301_MX6QPER2_MAIN;
WR4(sc, CCM_CCGR4, reg);
/* uarts, ssi, sdma */
reg = CCGR5_SDMA | CCGR5_SSI1 | CCGR5_SSI2 | CCGR5_SSI3 |
CCGR5_UART | CCGR5_UART_SERIAL;
WR4(sc, CCM_CCGR5, reg);
/* usdhc 1-4, usboh3 */
reg = CCGR6_USBOH3 | CCGR6_USDHC1 | CCGR6_USDHC2 |
CCGR6_USDHC3 | CCGR6_USDHC4;
WR4(sc, CCM_CCGR6, reg);
}
static int
ccm_detach(device_t dev)
{
struct ccm_softc *sc;
sc = device_get_softc(dev);
if (sc->mem_res != NULL)
bus_release_resource(dev, SYS_RES_MEMORY, 0, sc->mem_res);
return (0);
}
static int
ccm_attach(device_t dev)
{
struct ccm_softc *sc;
int err, rid;
uint32_t reg;
sc = device_get_softc(dev);
err = 0;
/* Allocate bus_space resources. */
rid = 0;
sc->mem_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY, &rid,
RF_ACTIVE);
if (sc->mem_res == NULL) {
device_printf(dev, "Cannot allocate memory resources\n");
err = ENXIO;
goto out;
}
ccm_sc = sc;
/*
* Configure the Low Power Mode setting to leave the ARM core power on
* when a WFI instruction is executed. This lets the MPCore timers and
* GIC continue to run, which is helpful when the only thing that can
* wake you up is an MPCore Private Timer interrupt delivered via GIC.
*
* XXX Based on the docs, setting CCM_CGPR_INT_MEM_CLK_LPM shouldn't be
* required when the LPM bits are set to LPM_RUN. But experimentally
* I've experienced a fairly rare lockup when not setting it. I was
* unable to prove conclusively that the lockup was related to power
* management or that this definitively fixes it. Revisit this.
*/
reg = RD4(sc, CCM_CGPR);
reg |= CCM_CGPR_INT_MEM_CLK_LPM;
WR4(sc, CCM_CGPR, reg);
reg = RD4(sc, CCM_CLPCR);
reg = (reg & ~CCM_CLPCR_LPM_MASK) | CCM_CLPCR_LPM_RUN;
WR4(sc, CCM_CLPCR, reg);
ccm_init_gates(sc);
err = 0;
out:
if (err != 0)
ccm_detach(dev);
return (err);
}
static int
ccm_probe(device_t dev)
{
if (!ofw_bus_status_okay(dev))
return (ENXIO);
if (ofw_bus_is_compatible(dev, "fsl,imx6q-ccm") == 0)
return (ENXIO);
device_set_desc(dev, "Freescale i.MX6 Clock Control Module");
return (BUS_PROBE_DEFAULT);
}
void
imx_ccm_ssi_configure(device_t _ssidev)
{
struct ccm_softc *sc;
uint32_t reg;
sc = ccm_sc;
/*
* Select PLL4 (Audio PLL) clock multiplexer as source.
* PLL output frequency = Fref * (DIV_SELECT + NUM/DENOM).
*/
reg = RD4(sc, CCM_CSCMR1);
reg &= ~(SSI_CLK_SEL_M << SSI1_CLK_SEL_S);
reg |= (SSI_CLK_SEL_PLL4 << SSI1_CLK_SEL_S);
reg &= ~(SSI_CLK_SEL_M << SSI2_CLK_SEL_S);
reg |= (SSI_CLK_SEL_PLL4 << SSI2_CLK_SEL_S);
reg &= ~(SSI_CLK_SEL_M << SSI3_CLK_SEL_S);
reg |= (SSI_CLK_SEL_PLL4 << SSI3_CLK_SEL_S);
WR4(sc, CCM_CSCMR1, reg);
/*
* Ensure we have set hardware-default values
* for pre and post dividers.
*/
/* SSI1 and SSI3 */
reg = RD4(sc, CCM_CS1CDR);
/* Divide by 2 */
reg &= ~(SSI_CLK_PODF_MASK << SSI1_CLK_PODF_SHIFT);
reg &= ~(SSI_CLK_PODF_MASK << SSI3_CLK_PODF_SHIFT);
reg |= (0x1 << SSI1_CLK_PODF_SHIFT);
reg |= (0x1 << SSI3_CLK_PODF_SHIFT);
/* Divide by 4 */
reg &= ~(SSI_CLK_PRED_MASK << SSI1_CLK_PRED_SHIFT);
reg &= ~(SSI_CLK_PRED_MASK << SSI3_CLK_PRED_SHIFT);
reg |= (0x3 << SSI1_CLK_PRED_SHIFT);
reg |= (0x3 << SSI3_CLK_PRED_SHIFT);
WR4(sc, CCM_CS1CDR, reg);
/* SSI2 */
reg = RD4(sc, CCM_CS2CDR);
/* Divide by 2 */
reg &= ~(SSI_CLK_PODF_MASK << SSI2_CLK_PODF_SHIFT);
reg |= (0x1 << SSI2_CLK_PODF_SHIFT);
/* Divide by 4 */
reg &= ~(SSI_CLK_PRED_MASK << SSI2_CLK_PRED_SHIFT);
reg |= (0x3 << SSI2_CLK_PRED_SHIFT);
WR4(sc, CCM_CS2CDR, reg);
}
void
imx_ccm_usb_enable(device_t _usbdev)
{
/*
* For imx6, the USBOH3 clock gate is bits 0-1 of CCGR6, so no need for
* shifting and masking here, just set the low-order two bits to ALWAYS.
*/
WR4(ccm_sc, CCM_CCGR6, RD4(ccm_sc, CCM_CCGR6) | CCGR_CLK_MODE_ALWAYS);
}
void
imx_ccm_usbphy_enable(device_t _phydev)
{
/*
* XXX Which unit?
* Right now it's not clear how to figure from fdt data which phy unit
* we're supposed to operate on. Until this is worked out, just enable
* both PHYs.
*/
#if 0
int phy_num, regoff;
phy_num = 0; /* XXX */
switch (phy_num) {
case 0:
regoff = 0;
break;
case 1:
regoff = 0x10;
break;
default:
device_printf(ccm_sc->dev, "Bad PHY number %u,\n",
phy_num);
return;
}
imx6_anatop_write_4(IMX6_ANALOG_CCM_PLL_USB1 + regoff,
IMX6_ANALOG_CCM_PLL_USB_ENABLE |
IMX6_ANALOG_CCM_PLL_USB_POWER |
IMX6_ANALOG_CCM_PLL_USB_EN_USB_CLKS);
#else
imx6_anatop_write_4(IMX6_ANALOG_CCM_PLL_USB1 + 0,
IMX6_ANALOG_CCM_PLL_USB_ENABLE |
IMX6_ANALOG_CCM_PLL_USB_POWER |
IMX6_ANALOG_CCM_PLL_USB_EN_USB_CLKS);
imx6_anatop_write_4(IMX6_ANALOG_CCM_PLL_USB1 + 0x10,
IMX6_ANALOG_CCM_PLL_USB_ENABLE |
IMX6_ANALOG_CCM_PLL_USB_POWER |
IMX6_ANALOG_CCM_PLL_USB_EN_USB_CLKS);
#endif
}
int
imx6_ccm_sata_enable(void)
{
uint32_t v;
int timeout;
/* Un-gate the sata controller. */
WR4(ccm_sc, CCM_CCGR5, RD4(ccm_sc, CCM_CCGR5) | CCGR5_SATA);
/* Power up the PLL that feeds ENET/SATA/PCI phys, wait for lock. */
v = RD4(ccm_sc, CCM_ANALOG_PLL_ENET);
v &= ~CCM_ANALOG_PLL_ENET_POWERDOWN;
WR4(ccm_sc, CCM_ANALOG_PLL_ENET, v);
for (timeout = 100000; timeout > 0; timeout--) {
if (RD4(ccm_sc, CCM_ANALOG_PLL_ENET) &
CCM_ANALOG_PLL_ENET_LOCK) {
break;
}
}
if (timeout <= 0) {
return ETIMEDOUT;
}
/* Enable the PLL, and enable its 100mhz output. */
v |= CCM_ANALOG_PLL_ENET_ENABLE;
v &= ~CCM_ANALOG_PLL_ENET_BYPASS;
WR4(ccm_sc, CCM_ANALOG_PLL_ENET, v);
v |= CCM_ANALOG_PLL_ENET_ENABLE_100M;
WR4(ccm_sc, CCM_ANALOG_PLL_ENET, v);
return 0;
}
uint32_t
imx_ccm_ecspi_hz(void)
{
return (60000000);
}
uint32_t
imx_ccm_ipg_hz(void)
{
return (66000000);
}
uint32_t
imx_ccm_perclk_hz(void)
{
return (66000000);
}
uint32_t
imx_ccm_sdhci_hz(void)
{
return (200000000);
}
uint32_t
imx_ccm_uart_hz(void)
{
return (80000000);
}
uint32_t
imx_ccm_ahb_hz(void)
{
return (132000000);
}
int
imx_ccm_pll_video_enable(void)
{
uint32_t reg;
int timeout;
/* Power down PLL */
reg = RD4(ccm_sc, CCM_ANALOG_PLL_VIDEO);
reg &= ~CCM_ANALOG_PLL_VIDEO_POWERDOWN;
WR4(ccm_sc, CCM_ANALOG_PLL_VIDEO, reg);
/*
* Fvideo = Fref * (37 + 11/12) / 2
* Fref = 24MHz, Fvideo = 455MHz
*/
reg &= ~CCM_ANALOG_PLL_VIDEO_POST_DIV_SELECT_MASK;
reg |= CCM_ANALOG_PLL_VIDEO_POST_DIV_2;
reg &= ~CCM_ANALOG_PLL_VIDEO_DIV_SELECT_MASK;
reg |= 37 << CCM_ANALOG_PLL_VIDEO_DIV_SELECT_SHIFT;
WR4(ccm_sc, CCM_ANALOG_PLL_VIDEO, reg);
WR4(ccm_sc, CCM_ANALOG_PLL_VIDEO_NUM, 11);
WR4(ccm_sc, CCM_ANALOG_PLL_VIDEO_DENOM, 12);
/* Power up and wait for PLL lock down */
reg = RD4(ccm_sc, CCM_ANALOG_PLL_VIDEO);
reg &= ~CCM_ANALOG_PLL_VIDEO_POWERDOWN;
WR4(ccm_sc, CCM_ANALOG_PLL_VIDEO, reg);
for (timeout = 100000; timeout > 0; timeout--) {
if (RD4(ccm_sc, CCM_ANALOG_PLL_VIDEO) &
CCM_ANALOG_PLL_VIDEO_LOCK) {
break;
}
}
if (timeout <= 0) {
return ETIMEDOUT;
}
/* Enable the PLL */
reg |= CCM_ANALOG_PLL_VIDEO_ENABLE;
reg &= ~CCM_ANALOG_PLL_VIDEO_BYPASS;
WR4(ccm_sc, CCM_ANALOG_PLL_VIDEO, reg);
return (0);
}
void
imx_ccm_ipu_enable(int ipu)
{
struct ccm_softc *sc;
uint32_t reg;
sc = ccm_sc;
reg = RD4(sc, CCM_CCGR3);
if (ipu == 1)
reg |= CCGR3_IPU1_IPU | CCGR3_IPU1_DI0;
else
reg |= CCGR3_IPU2_IPU | CCGR3_IPU2_DI0;
WR4(sc, CCM_CCGR3, reg);
/* Set IPU1_DI0 clock to source from PLL5 and divide it by 3 */
reg = RD4(sc, CCM_CHSCCDR);
reg &= ~(CHSCCDR_IPU1_DI0_PRE_CLK_SEL_MASK |
CHSCCDR_IPU1_DI0_PODF_MASK | CHSCCDR_IPU1_DI0_CLK_SEL_MASK);
reg |= (CHSCCDR_PODF_DIVIDE_BY_3 << CHSCCDR_IPU1_DI0_PODF_SHIFT);
reg |= (CHSCCDR_IPU_PRE_CLK_PLL5 << CHSCCDR_IPU1_DI0_PRE_CLK_SEL_SHIFT);
WR4(sc, CCM_CHSCCDR, reg);
reg |= (CHSCCDR_CLK_SEL_PREMUXED << CHSCCDR_IPU1_DI0_CLK_SEL_SHIFT);
WR4(sc, CCM_CHSCCDR, reg);
}
uint32_t
imx_ccm_ipu_hz(void)
{
return (455000000 / 3);
}
void
imx_ccm_hdmi_enable(void)
{
struct ccm_softc *sc;
uint32_t reg;
sc = ccm_sc;
reg = RD4(sc, CCM_CCGR2);
reg |= CCGR2_HDMI_TX | CCGR2_HDMI_TX_ISFR;
WR4(sc, CCM_CCGR2, reg);
}
uint32_t
imx_ccm_get_cacrr(void)
{
return (RD4(ccm_sc, CCM_CACCR));
}
void
imx_ccm_set_cacrr(uint32_t divisor)
{
WR4(ccm_sc, CCM_CACCR, divisor);
}
static device_method_t ccm_methods[] = {
/* Device interface */
DEVMETHOD(device_probe, ccm_probe),
DEVMETHOD(device_attach, ccm_attach),
DEVMETHOD(device_detach, ccm_detach),
DEVMETHOD_END
};
static driver_t ccm_driver = {
"ccm",
ccm_methods,
sizeof(struct ccm_softc)
};
static devclass_t ccm_devclass;
EARLY_DRIVER_MODULE(ccm, simplebus, ccm_driver, ccm_devclass, 0, 0,
BUS_PASS_CPU + BUS_PASS_ORDER_EARLY);