mach-imx6q.c 8.3 KB
Newer Older
1
/*
2
 * Copyright 2011-2013 Freescale Semiconductor, Inc.
3 4 5 6 7 8 9 10 11 12
 * Copyright 2011 Linaro Ltd.
 *
 * The code contained herein is licensed under the GNU General Public
 * License. You may obtain a copy of the GNU General Public License
 * Version 2 or later at the following locations:
 *
 * http://www.opensource.org/licenses/gpl-license.html
 * http://www.gnu.org/copyleft/gpl.html
 */

13 14
#include <linux/clk.h>
#include <linux/clkdev.h>
15
#include <linux/cpu.h>
16
#include <linux/delay.h>
R
Robert Lee 已提交
17
#include <linux/export.h>
18
#include <linux/init.h>
19
#include <linux/io.h>
20
#include <linux/irq.h>
21
#include <linux/irqchip.h>
22
#include <linux/of.h>
23
#include <linux/of_address.h>
24 25
#include <linux/of_irq.h>
#include <linux/of_platform.h>
26
#include <linux/pm_opp.h>
27
#include <linux/pci.h>
28
#include <linux/phy.h>
29
#include <linux/reboot.h>
30
#include <linux/regmap.h>
31
#include <linux/micrel_phy.h>
32
#include <linux/mfd/syscon.h>
33
#include <linux/mfd/syscon/imx6q-iomuxc-gpr.h>
34
#include <asm/mach/arch.h>
35
#include <asm/mach/map.h>
36
#include <asm/system_misc.h>
37

38
#include "common.h"
39
#include "cpuidle.h"
40
#include "hardware.h"
R
Robert Lee 已提交
41

42 43 44
/* For imx6q sabrelite board: set KSZ9021RN RGMII pad skew */
static int ksz9021rn_phy_fixup(struct phy_device *phydev)
{
45
	if (IS_BUILTIN(CONFIG_PHYLIB)) {
46
		/* min rx data delay */
47 48 49
		phy_write(phydev, MICREL_KSZ9021_EXTREG_CTRL,
			0x8000 | MICREL_KSZ9021_RGMII_RX_DATA_PAD_SCEW);
		phy_write(phydev, MICREL_KSZ9021_EXTREG_DATA_WRITE, 0x0000);
50

51
		/* max rx/tx clock delay, min rx/tx control delay */
52 53 54 55 56
		phy_write(phydev, MICREL_KSZ9021_EXTREG_CTRL,
			0x8000 | MICREL_KSZ9021_RGMII_CLK_CTRL_PAD_SCEW);
		phy_write(phydev, MICREL_KSZ9021_EXTREG_DATA_WRITE, 0xf0f0);
		phy_write(phydev, MICREL_KSZ9021_EXTREG_CTRL,
			MICREL_KSZ9021_RGMII_CLK_CTRL_PAD_SCEW);
57
	}
58 59 60 61

	return 0;
}

62
static void mmd_write_reg(struct phy_device *dev, int device, int reg, int val)
63
{
64 65 66 67
	phy_write(dev, 0x0d, device);
	phy_write(dev, 0x0e, reg);
	phy_write(dev, 0x0d, (1 << 14) | device);
	phy_write(dev, 0x0e, val);
68 69
}

70
static int ksz9031rn_phy_fixup(struct phy_device *dev)
71
{
72 73 74 75 76 77 78 79 80
	/*
	 * min rx data delay, max rx/tx clock delay,
	 * min rx/tx control delay
	 */
	mmd_write_reg(dev, 2, 4, 0);
	mmd_write_reg(dev, 2, 5, 0);
	mmd_write_reg(dev, 2, 8, 0x003ff);

	return 0;
81 82
}

83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
/*
 * fixup for PLX PEX8909 bridge to configure GPIO1-7 as output High
 * as they are used for slots1-7 PERST#
 */
static void ventana_pciesw_early_fixup(struct pci_dev *dev)
{
	u32 dw;

	if (!of_machine_is_compatible("gw,ventana"))
		return;

	if (dev->devfn != 0)
		return;

	pci_read_config_dword(dev, 0x62c, &dw);
	dw |= 0xaaa8; // GPIO1-7 outputs
	pci_write_config_dword(dev, 0x62c, dw);

	pci_read_config_dword(dev, 0x644, &dw);
	dw |= 0xfe;   // GPIO1-7 output high
	pci_write_config_dword(dev, 0x644, dw);

	msleep(100);
}
DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_PLX, 0x8609, ventana_pciesw_early_fixup);
DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_PLX, 0x8606, ventana_pciesw_early_fixup);
DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_PLX, 0x8604, ventana_pciesw_early_fixup);

111
static int ar8031_phy_fixup(struct phy_device *dev)
112
{
113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
	u16 val;

	/* To enable AR8031 output a 125MHz clk from CLK_25M */
	phy_write(dev, 0xd, 0x7);
	phy_write(dev, 0xe, 0x8016);
	phy_write(dev, 0xd, 0x4007);

	val = phy_read(dev, 0xe);
	val &= 0xffe3;
	val |= 0x18;
	phy_write(dev, 0xe, val);

	/* introduce tx clock delay */
	phy_write(dev, 0x1d, 0x5);
	val = phy_read(dev, 0x1e);
	val |= 0x0100;
	phy_write(dev, 0x1e, val);

	return 0;
132 133
}

134 135
#define PHY_ID_AR8031	0x004dd074

136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168
static int ar8035_phy_fixup(struct phy_device *dev)
{
	u16 val;

	/* Ar803x phy SmartEEE feature cause link status generates glitch,
	 * which cause ethernet link down/up issue, so disable SmartEEE
	 */
	phy_write(dev, 0xd, 0x3);
	phy_write(dev, 0xe, 0x805d);
	phy_write(dev, 0xd, 0x4003);

	val = phy_read(dev, 0xe);
	phy_write(dev, 0xe, val & ~(1 << 8));

	/*
	 * Enable 125MHz clock from CLK_25M on the AR8031.  This
	 * is fed in to the IMX6 on the ENET_REF_CLK (V22) pad.
	 * Also, introduce a tx clock delay.
	 *
	 * This is the same as is the AR8031 fixup.
	 */
	ar8031_phy_fixup(dev);

	/*check phy power*/
	val = phy_read(dev, 0x0);
	if (val & BMCR_PDOWN)
		phy_write(dev, 0x0, val & ~BMCR_PDOWN);

	return 0;
}

#define PHY_ID_AR8035 0x004dd072

169
static void __init imx6q_enet_phy_init(void)
170
{
171
	if (IS_BUILTIN(CONFIG_PHYLIB)) {
172
		phy_register_fixup_for_uid(PHY_ID_KSZ9021, MICREL_PHY_ID_MASK,
173
				ksz9021rn_phy_fixup);
174 175
		phy_register_fixup_for_uid(PHY_ID_KSZ9031, MICREL_PHY_ID_MASK,
				ksz9031rn_phy_fixup);
176 177
		phy_register_fixup_for_uid(PHY_ID_AR8031, 0xffffffff,
				ar8031_phy_fixup);
178 179
		phy_register_fixup_for_uid(PHY_ID_AR8035, 0xffffffef,
				ar8035_phy_fixup);
180
	}
181 182
}

183 184
static void __init imx6q_1588_init(void)
{
185 186 187
	struct device_node *np;
	struct clk *ptp_clk;
	struct clk *enet_ref;
188
	struct regmap *gpr;
189
	u32 clksel;
190

191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215
	np = of_find_compatible_node(NULL, NULL, "fsl,imx6q-fec");
	if (!np) {
		pr_warn("%s: failed to find fec node\n", __func__);
		return;
	}

	ptp_clk = of_clk_get(np, 2);
	if (IS_ERR(ptp_clk)) {
		pr_warn("%s: failed to get ptp clock\n", __func__);
		goto put_node;
	}

	enet_ref = clk_get_sys(NULL, "enet_ref");
	if (IS_ERR(enet_ref)) {
		pr_warn("%s: failed to get enet clock\n", __func__);
		goto put_ptp_clk;
	}

	/*
	 * If enet_ref from ANATOP/CCM is the PTP clock source, we need to
	 * set bit IOMUXC_GPR1[21].  Or the PTP clock must be from pad
	 * (external OSC), and we need to clear the bit.
	 */
	clksel = ptp_clk == enet_ref ? IMX6Q_GPR1_ENET_CLK_SEL_ANATOP :
				       IMX6Q_GPR1_ENET_CLK_SEL_PAD;
216 217
	gpr = syscon_regmap_lookup_by_compatible("fsl,imx6q-iomuxc-gpr");
	if (!IS_ERR(gpr))
218 219
		regmap_update_bits(gpr, IOMUXC_GPR1,
				IMX6Q_GPR1_ENET_CLK_SEL_MASK,
220
				clksel);
221 222 223
	else
		pr_err("failed to find fsl,imx6q-iomux-gpr regmap\n");

224 225 226 227 228
	clk_put(enet_ref);
put_ptp_clk:
	clk_put(ptp_clk);
put_node:
	of_node_put(np);
229
}
230

231 232
static void __init imx6q_init_machine(void)
{
233 234
	struct device *parent;

235
	imx_print_silicon_rev(cpu_is_imx6dl() ? "i.MX6DL" : "i.MX6Q",
236
			      imx_get_soc_revision());
237

238 239
	mxc_arch_reset_init_dt();

240 241 242 243
	parent = imx_soc_device_init();
	if (parent == NULL)
		pr_warn("failed to initialize soc device\n");

244
	imx6q_enet_phy_init();
245

246
	of_platform_populate(NULL, of_default_bus_match_table, NULL, parent);
247

248
	imx_anatop_init();
249
	cpu_is_imx6q() ?  imx6q_pm_init() : imx6dl_pm_init();
250
	imx6q_1588_init();
251 252
}

253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277
#define OCOTP_CFG3			0x440
#define OCOTP_CFG3_SPEED_SHIFT		16
#define OCOTP_CFG3_SPEED_1P2GHZ		0x3

static void __init imx6q_opp_check_1p2ghz(struct device *cpu_dev)
{
	struct device_node *np;
	void __iomem *base;
	u32 val;

	np = of_find_compatible_node(NULL, NULL, "fsl,imx6q-ocotp");
	if (!np) {
		pr_warn("failed to find ocotp node\n");
		return;
	}

	base = of_iomap(np, 0);
	if (!base) {
		pr_warn("failed to map ocotp\n");
		goto put_node;
	}

	val = readl_relaxed(base + OCOTP_CFG3);
	val >>= OCOTP_CFG3_SPEED_SHIFT;
	if ((val & 0x3) != OCOTP_CFG3_SPEED_1P2GHZ)
278
		if (dev_pm_opp_disable(cpu_dev, 1200000000))
279 280 281 282 283 284
			pr_warn("failed to disable 1.2 GHz OPP\n");

put_node:
	of_node_put(np);
}

285
static void __init imx6q_opp_init(void)
286 287
{
	struct device_node *np;
288
	struct device *cpu_dev = get_cpu_device(0);
289

290 291 292 293
	if (!cpu_dev) {
		pr_warn("failed to get cpu0 device\n");
		return;
	}
294
	np = of_node_get(cpu_dev->of_node);
295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310
	if (!np) {
		pr_warn("failed to find cpu0 node\n");
		return;
	}

	if (of_init_opp_table(cpu_dev)) {
		pr_warn("failed to init OPP table\n");
		goto put_node;
	}

	imx6q_opp_check_1p2ghz(cpu_dev);

put_node:
	of_node_put(np);
}

311
static struct platform_device imx6q_cpufreq_pdev = {
312 313 314
	.name = "imx6q-cpufreq",
};

R
Robert Lee 已提交
315 316
static void __init imx6q_init_late(void)
{
317 318 319 320
	/*
	 * WAIT mode is broken on TO 1.0 and 1.1, so there is no point
	 * to run cpuidle on them.
	 */
321
	if (imx_get_soc_revision() > IMX_CHIP_REVISION_1_1)
322
		imx6q_cpuidle_init();
323 324

	if (IS_ENABLED(CONFIG_ARM_IMX6Q_CPUFREQ)) {
325
		imx6q_opp_init();
326 327
		platform_device_register(&imx6q_cpufreq_pdev);
	}
R
Robert Lee 已提交
328 329
}

330 331
static void __init imx6q_map_io(void)
{
332
	debug_ll_io_init();
333 334 335 336 337
	imx_scu_map_io();
}

static void __init imx6q_init_irq(void)
{
338
	imx_init_revision_from_anatop();
339
	imx_init_l2cache();
340 341
	imx_src_init();
	imx_gpc_init();
342
	irqchip_init();
343 344
}

345
static const char *imx6q_dt_compat[] __initconst = {
S
Shawn Guo 已提交
346
	"fsl,imx6dl",
347
	"fsl,imx6q",
348 349 350
	NULL,
};

S
Shawn Guo 已提交
351
DT_MACHINE_START(IMX6Q, "Freescale i.MX6 Quad/DualLite (Device Tree)")
352
	.smp		= smp_ops(imx_smp_ops),
353 354 355
	.map_io		= imx6q_map_io,
	.init_irq	= imx6q_init_irq,
	.init_machine	= imx6q_init_machine,
R
Robert Lee 已提交
356
	.init_late      = imx6q_init_late,
357
	.dt_compat	= imx6q_dt_compat,
358
	.restart	= mxc_restart,
359
MACHINE_END