Linux Audio

Check our new training course

Loading...
Note: File does not exist in v3.5.6.
  1// SPDX-License-Identifier: GPL-2.0
  2/*
  3 *  Nintendo 64 init.
  4 *
  5 *  Copyright (C) 2021	Lauri Kasanen
  6 */
  7#include <linux/init.h>
  8#include <linux/ioport.h>
  9#include <linux/irq.h>
 10#include <linux/memblock.h>
 11#include <linux/platform_device.h>
 12#include <linux/platform_data/simplefb.h>
 13#include <linux/string.h>
 14
 15#include <asm/bootinfo.h>
 16#include <asm/fw/fw.h>
 17#include <asm/time.h>
 18
 19#define IO_MEM_RESOURCE_START	0UL
 20#define IO_MEM_RESOURCE_END	0x1fffffffUL
 21
 22/*
 23 * System-specifc irq names for clarity
 24 */
 25#define MIPS_CPU_IRQ(x)		(MIPS_CPU_IRQ_BASE + (x))
 26#define MIPS_SOFTINT0_IRQ	MIPS_CPU_IRQ(0)
 27#define MIPS_SOFTINT1_IRQ	MIPS_CPU_IRQ(1)
 28#define RCP_IRQ			MIPS_CPU_IRQ(2)
 29#define CART_IRQ		MIPS_CPU_IRQ(3)
 30#define PRENMI_IRQ		MIPS_CPU_IRQ(4)
 31#define RDBR_IRQ		MIPS_CPU_IRQ(5)
 32#define RDBW_IRQ		MIPS_CPU_IRQ(6)
 33#define TIMER_IRQ		MIPS_CPU_IRQ(7)
 34
 35static void __init iomem_resource_init(void)
 36{
 37	iomem_resource.start = IO_MEM_RESOURCE_START;
 38	iomem_resource.end = IO_MEM_RESOURCE_END;
 39}
 40
 41const char *get_system_type(void)
 42{
 43	return "Nintendo 64";
 44}
 45
 46void __init prom_init(void)
 47{
 48	fw_init_cmdline();
 49}
 50
 51#define W 320
 52#define H 240
 53#define REG_BASE ((u32 *) CKSEG1ADDR(0x4400000))
 54
 55static void __init n64rdp_write_reg(const u8 reg, const u32 value)
 56{
 57	__raw_writel(value, REG_BASE + reg);
 58}
 59
 60#undef REG_BASE
 61
 62static const u32 ntsc_320[] __initconst = {
 63	0x00013212, 0x00000000, 0x00000140, 0x00000200,
 64	0x00000000, 0x03e52239, 0x0000020d, 0x00000c15,
 65	0x0c150c15, 0x006c02ec, 0x002501ff, 0x000e0204,
 66	0x00000200, 0x00000400
 67};
 68
 69#define MI_REG_BASE 0x4300000
 70#define NUM_MI_REGS 4
 71#define AI_REG_BASE 0x4500000
 72#define NUM_AI_REGS 6
 73#define PI_REG_BASE 0x4600000
 74#define NUM_PI_REGS 5
 75#define SI_REG_BASE 0x4800000
 76#define NUM_SI_REGS 7
 77
 78static int __init n64_platform_init(void)
 79{
 80	static const char simplefb_resname[] = "FB";
 81	static const struct simplefb_platform_data mode = {
 82		.width = W,
 83		.height = H,
 84		.stride = W * 2,
 85		.format = "r5g5b5a1"
 86	};
 87	struct resource res[3];
 88	void *orig;
 89	unsigned long phys;
 90	unsigned i;
 91
 92	memset(res, 0, sizeof(struct resource) * 3);
 93	res[0].flags = IORESOURCE_MEM;
 94	res[0].start = MI_REG_BASE;
 95	res[0].end = MI_REG_BASE + NUM_MI_REGS * 4 - 1;
 96
 97	res[1].flags = IORESOURCE_MEM;
 98	res[1].start = AI_REG_BASE;
 99	res[1].end = AI_REG_BASE + NUM_AI_REGS * 4 - 1;
100
101	res[2].flags = IORESOURCE_IRQ;
102	res[2].start = RCP_IRQ;
103	res[2].end = RCP_IRQ;
104
105	platform_device_register_simple("n64audio", -1, res, 3);
106
107	memset(&res[0], 0, sizeof(res[0]));
108	res[0].flags = IORESOURCE_MEM;
109	res[0].start = PI_REG_BASE;
110	res[0].end = PI_REG_BASE + NUM_PI_REGS * 4 - 1;
111
112	platform_device_register_simple("n64cart", -1, res, 1);
113
114	memset(&res[0], 0, sizeof(res[0]));
115	res[0].flags = IORESOURCE_MEM;
116	res[0].start = SI_REG_BASE;
117	res[0].end = SI_REG_BASE + NUM_SI_REGS * 4 - 1;
118
119	platform_device_register_simple("n64joy", -1, res, 1);
120
121	/* The framebuffer needs 64-byte alignment */
122	orig = kzalloc(W * H * 2 + 63, GFP_DMA | GFP_KERNEL);
123	if (!orig)
124		return -ENOMEM;
125	phys = virt_to_phys(orig);
126	phys += 63;
127	phys &= ~63;
128
129	for (i = 0; i < ARRAY_SIZE(ntsc_320); i++) {
130		if (i == 1)
131			n64rdp_write_reg(i, phys);
132		else
133			n64rdp_write_reg(i, ntsc_320[i]);
134	}
135
136	/* setup IORESOURCE_MEM as framebuffer memory */
137	memset(&res[0], 0, sizeof(res[0]));
138	res[0].flags = IORESOURCE_MEM;
139	res[0].name = simplefb_resname;
140	res[0].start = phys;
141	res[0].end = phys + W * H * 2 - 1;
142
143	platform_device_register_resndata(NULL, "simple-framebuffer", 0,
144					  &res[0], 1, &mode, sizeof(mode));
145
146	return 0;
147}
148
149#undef W
150#undef H
151
152arch_initcall(n64_platform_init);
153
154void __init plat_mem_setup(void)
155{
156	iomem_resource_init();
157	memblock_add(0x0, 8 * 1024 * 1024); /* Bootloader blocks the 4mb config */
158}
159
160void __init plat_time_init(void)
161{
162	/* 93.75 MHz cpu, count register runs at half rate */
163	mips_hpt_frequency = 93750000 / 2;
164}