162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Setup for the SMSC FDC37C93xAPM 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2003 Sean McGoogan (Sean.McGoogan@superh.com) 662306a36Sopenharmony_ci * Copyright (C) 2003, 2004 SuperH, Inc. 762306a36Sopenharmony_ci * Copyright (C) 2004, 2005 Paul Mundt 862306a36Sopenharmony_ci * 962306a36Sopenharmony_ci * SuperH SH4-202 MicroDev board support. 1062306a36Sopenharmony_ci */ 1162306a36Sopenharmony_ci#include <linux/init.h> 1262306a36Sopenharmony_ci#include <linux/ioport.h> 1362306a36Sopenharmony_ci#include <linux/io.h> 1462306a36Sopenharmony_ci#include <linux/err.h> 1562306a36Sopenharmony_ci#include <mach/microdev.h> 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci#define SMSC_CONFIG_PORT_ADDR (0x3F0) 1862306a36Sopenharmony_ci#define SMSC_INDEX_PORT_ADDR SMSC_CONFIG_PORT_ADDR 1962306a36Sopenharmony_ci#define SMSC_DATA_PORT_ADDR (SMSC_INDEX_PORT_ADDR + 1) 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci#define SMSC_ENTER_CONFIG_KEY 0x55 2262306a36Sopenharmony_ci#define SMSC_EXIT_CONFIG_KEY 0xaa 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ci#define SMCS_LOGICAL_DEV_INDEX 0x07 /* Logical Device Number */ 2562306a36Sopenharmony_ci#define SMSC_DEVICE_ID_INDEX 0x20 /* Device ID */ 2662306a36Sopenharmony_ci#define SMSC_DEVICE_REV_INDEX 0x21 /* Device Revision */ 2762306a36Sopenharmony_ci#define SMSC_ACTIVATE_INDEX 0x30 /* Activate */ 2862306a36Sopenharmony_ci#define SMSC_PRIMARY_BASE_INDEX 0x60 /* Primary Base Address */ 2962306a36Sopenharmony_ci#define SMSC_SECONDARY_BASE_INDEX 0x62 /* Secondary Base Address */ 3062306a36Sopenharmony_ci#define SMSC_PRIMARY_INT_INDEX 0x70 /* Primary Interrupt Select */ 3162306a36Sopenharmony_ci#define SMSC_SECONDARY_INT_INDEX 0x72 /* Secondary Interrupt Select */ 3262306a36Sopenharmony_ci#define SMSC_HDCS0_INDEX 0xf0 /* HDCS0 Address Decoder */ 3362306a36Sopenharmony_ci#define SMSC_HDCS1_INDEX 0xf1 /* HDCS1 Address Decoder */ 3462306a36Sopenharmony_ci 3562306a36Sopenharmony_ci#define SMSC_IDE1_DEVICE 1 /* IDE #1 logical device */ 3662306a36Sopenharmony_ci#define SMSC_IDE2_DEVICE 2 /* IDE #2 logical device */ 3762306a36Sopenharmony_ci#define SMSC_PARALLEL_DEVICE 3 /* Parallel Port logical device */ 3862306a36Sopenharmony_ci#define SMSC_SERIAL1_DEVICE 4 /* Serial #1 logical device */ 3962306a36Sopenharmony_ci#define SMSC_SERIAL2_DEVICE 5 /* Serial #2 logical device */ 4062306a36Sopenharmony_ci#define SMSC_KEYBOARD_DEVICE 7 /* Keyboard logical device */ 4162306a36Sopenharmony_ci#define SMSC_CONFIG_REGISTERS 8 /* Configuration Registers (Aux I/O) */ 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ci#define SMSC_READ_INDEXED(index) ({ \ 4462306a36Sopenharmony_ci outb((index), SMSC_INDEX_PORT_ADDR); \ 4562306a36Sopenharmony_ci inb(SMSC_DATA_PORT_ADDR); }) 4662306a36Sopenharmony_ci#define SMSC_WRITE_INDEXED(val, index) ({ \ 4762306a36Sopenharmony_ci outb((index), SMSC_INDEX_PORT_ADDR); \ 4862306a36Sopenharmony_ci outb((val), SMSC_DATA_PORT_ADDR); }) 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_ci#define IDE1_PRIMARY_BASE 0x01f0 /* Task File Registe base for IDE #1 */ 5162306a36Sopenharmony_ci#define IDE1_SECONDARY_BASE 0x03f6 /* Miscellaneous AT registers for IDE #1 */ 5262306a36Sopenharmony_ci#define IDE2_PRIMARY_BASE 0x0170 /* Task File Registe base for IDE #2 */ 5362306a36Sopenharmony_ci#define IDE2_SECONDARY_BASE 0x0376 /* Miscellaneous AT registers for IDE #2 */ 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_ci#define SERIAL1_PRIMARY_BASE 0x03f8 5662306a36Sopenharmony_ci#define SERIAL2_PRIMARY_BASE 0x02f8 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_ci#define MSB(x) ( (x) >> 8 ) 5962306a36Sopenharmony_ci#define LSB(x) ( (x) & 0xff ) 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_ci /* General-Purpose base address on CPU-board FPGA */ 6262306a36Sopenharmony_ci#define MICRODEV_FPGA_GP_BASE 0xa6100000ul 6362306a36Sopenharmony_ci 6462306a36Sopenharmony_cistatic int __init smsc_superio_setup(void) 6562306a36Sopenharmony_ci{ 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci unsigned char devid, devrev; 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_ci /* Initially the chip is in run state */ 7062306a36Sopenharmony_ci /* Put it into configuration state */ 7162306a36Sopenharmony_ci outb(SMSC_ENTER_CONFIG_KEY, SMSC_CONFIG_PORT_ADDR); 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci /* Read device ID info */ 7462306a36Sopenharmony_ci devid = SMSC_READ_INDEXED(SMSC_DEVICE_ID_INDEX); 7562306a36Sopenharmony_ci devrev = SMSC_READ_INDEXED(SMSC_DEVICE_REV_INDEX); 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_ci if ((devid == 0x30) && (devrev == 0x01)) 7862306a36Sopenharmony_ci printk("SMSC FDC37C93xAPM SuperIO device detected\n"); 7962306a36Sopenharmony_ci else 8062306a36Sopenharmony_ci return -ENODEV; 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_ci /* Select the keyboard device */ 8362306a36Sopenharmony_ci SMSC_WRITE_INDEXED(SMSC_KEYBOARD_DEVICE, SMCS_LOGICAL_DEV_INDEX); 8462306a36Sopenharmony_ci /* enable it */ 8562306a36Sopenharmony_ci SMSC_WRITE_INDEXED(1, SMSC_ACTIVATE_INDEX); 8662306a36Sopenharmony_ci /* enable the interrupts */ 8762306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MICRODEV_FPGA_IRQ_KEYBOARD, SMSC_PRIMARY_INT_INDEX); 8862306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MICRODEV_FPGA_IRQ_MOUSE, SMSC_SECONDARY_INT_INDEX); 8962306a36Sopenharmony_ci 9062306a36Sopenharmony_ci /* Select the Serial #1 device */ 9162306a36Sopenharmony_ci SMSC_WRITE_INDEXED(SMSC_SERIAL1_DEVICE, SMCS_LOGICAL_DEV_INDEX); 9262306a36Sopenharmony_ci /* enable it */ 9362306a36Sopenharmony_ci SMSC_WRITE_INDEXED(1, SMSC_ACTIVATE_INDEX); 9462306a36Sopenharmony_ci /* program with port addresses */ 9562306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MSB(SERIAL1_PRIMARY_BASE), SMSC_PRIMARY_BASE_INDEX+0); 9662306a36Sopenharmony_ci SMSC_WRITE_INDEXED(LSB(SERIAL1_PRIMARY_BASE), SMSC_PRIMARY_BASE_INDEX+1); 9762306a36Sopenharmony_ci SMSC_WRITE_INDEXED(0x00, SMSC_HDCS0_INDEX); 9862306a36Sopenharmony_ci /* enable the interrupts */ 9962306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MICRODEV_FPGA_IRQ_SERIAL1, SMSC_PRIMARY_INT_INDEX); 10062306a36Sopenharmony_ci 10162306a36Sopenharmony_ci /* Select the Serial #2 device */ 10262306a36Sopenharmony_ci SMSC_WRITE_INDEXED(SMSC_SERIAL2_DEVICE, SMCS_LOGICAL_DEV_INDEX); 10362306a36Sopenharmony_ci /* enable it */ 10462306a36Sopenharmony_ci SMSC_WRITE_INDEXED(1, SMSC_ACTIVATE_INDEX); 10562306a36Sopenharmony_ci /* program with port addresses */ 10662306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MSB(SERIAL2_PRIMARY_BASE), SMSC_PRIMARY_BASE_INDEX+0); 10762306a36Sopenharmony_ci SMSC_WRITE_INDEXED(LSB(SERIAL2_PRIMARY_BASE), SMSC_PRIMARY_BASE_INDEX+1); 10862306a36Sopenharmony_ci SMSC_WRITE_INDEXED(0x00, SMSC_HDCS0_INDEX); 10962306a36Sopenharmony_ci /* enable the interrupts */ 11062306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MICRODEV_FPGA_IRQ_SERIAL2, SMSC_PRIMARY_INT_INDEX); 11162306a36Sopenharmony_ci 11262306a36Sopenharmony_ci /* Select the IDE#1 device */ 11362306a36Sopenharmony_ci SMSC_WRITE_INDEXED(SMSC_IDE1_DEVICE, SMCS_LOGICAL_DEV_INDEX); 11462306a36Sopenharmony_ci /* enable it */ 11562306a36Sopenharmony_ci SMSC_WRITE_INDEXED(1, SMSC_ACTIVATE_INDEX); 11662306a36Sopenharmony_ci /* program with port addresses */ 11762306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MSB(IDE1_PRIMARY_BASE), SMSC_PRIMARY_BASE_INDEX+0); 11862306a36Sopenharmony_ci SMSC_WRITE_INDEXED(LSB(IDE1_PRIMARY_BASE), SMSC_PRIMARY_BASE_INDEX+1); 11962306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MSB(IDE1_SECONDARY_BASE), SMSC_SECONDARY_BASE_INDEX+0); 12062306a36Sopenharmony_ci SMSC_WRITE_INDEXED(LSB(IDE1_SECONDARY_BASE), SMSC_SECONDARY_BASE_INDEX+1); 12162306a36Sopenharmony_ci SMSC_WRITE_INDEXED(0x0c, SMSC_HDCS0_INDEX); 12262306a36Sopenharmony_ci SMSC_WRITE_INDEXED(0x00, SMSC_HDCS1_INDEX); 12362306a36Sopenharmony_ci /* select the interrupt */ 12462306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MICRODEV_FPGA_IRQ_IDE1, SMSC_PRIMARY_INT_INDEX); 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_ci /* Select the IDE#2 device */ 12762306a36Sopenharmony_ci SMSC_WRITE_INDEXED(SMSC_IDE2_DEVICE, SMCS_LOGICAL_DEV_INDEX); 12862306a36Sopenharmony_ci /* enable it */ 12962306a36Sopenharmony_ci SMSC_WRITE_INDEXED(1, SMSC_ACTIVATE_INDEX); 13062306a36Sopenharmony_ci /* program with port addresses */ 13162306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MSB(IDE2_PRIMARY_BASE), SMSC_PRIMARY_BASE_INDEX+0); 13262306a36Sopenharmony_ci SMSC_WRITE_INDEXED(LSB(IDE2_PRIMARY_BASE), SMSC_PRIMARY_BASE_INDEX+1); 13362306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MSB(IDE2_SECONDARY_BASE), SMSC_SECONDARY_BASE_INDEX+0); 13462306a36Sopenharmony_ci SMSC_WRITE_INDEXED(LSB(IDE2_SECONDARY_BASE), SMSC_SECONDARY_BASE_INDEX+1); 13562306a36Sopenharmony_ci /* select the interrupt */ 13662306a36Sopenharmony_ci SMSC_WRITE_INDEXED(MICRODEV_FPGA_IRQ_IDE2, SMSC_PRIMARY_INT_INDEX); 13762306a36Sopenharmony_ci 13862306a36Sopenharmony_ci /* Select the configuration registers */ 13962306a36Sopenharmony_ci SMSC_WRITE_INDEXED(SMSC_CONFIG_REGISTERS, SMCS_LOGICAL_DEV_INDEX); 14062306a36Sopenharmony_ci /* enable the appropriate GPIO pins for IDE functionality: 14162306a36Sopenharmony_ci * bit[0] In/Out 1==input; 0==output 14262306a36Sopenharmony_ci * bit[1] Polarity 1==invert; 0==no invert 14362306a36Sopenharmony_ci * bit[2] Int Enb #1 1==Enable Combined IRQ #1; 0==disable 14462306a36Sopenharmony_ci * bit[3:4] Function Select 00==original; 01==Alternate Function #1 14562306a36Sopenharmony_ci */ 14662306a36Sopenharmony_ci SMSC_WRITE_INDEXED(0x00, 0xc2); /* GP42 = nIDE1_OE */ 14762306a36Sopenharmony_ci SMSC_WRITE_INDEXED(0x01, 0xc5); /* GP45 = IDE1_IRQ */ 14862306a36Sopenharmony_ci SMSC_WRITE_INDEXED(0x00, 0xc6); /* GP46 = nIOROP */ 14962306a36Sopenharmony_ci SMSC_WRITE_INDEXED(0x00, 0xc7); /* GP47 = nIOWOP */ 15062306a36Sopenharmony_ci SMSC_WRITE_INDEXED(0x08, 0xe8); /* GP20 = nIDE2_OE */ 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci /* Exit the configuration state */ 15362306a36Sopenharmony_ci outb(SMSC_EXIT_CONFIG_KEY, SMSC_CONFIG_PORT_ADDR); 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ci return 0; 15662306a36Sopenharmony_ci} 15762306a36Sopenharmony_cidevice_initcall(smsc_superio_setup); 158