2 * Copyright (c) 2009,2010 One Laptop per Child
4 * This program is free software. You can redistribute it and/or
5 * modify it under the terms of version 2 of the GNU General Public
6 * License as published by the Free Software Foundation.
9 #include <linux/acpi.h>
10 #include <linux/pci.h>
11 #include <linux/gpio.h>
14 /* TODO: this eventually belongs in linux/vx855.h */
15 #define NR_VX855_GPI 14
16 #define NR_VX855_GPO 13
17 #define NR_VX855_GPIO 15
19 #define VX855_GPI(n) (n)
20 #define VX855_GPO(n) (NR_VX855_GPI + (n))
21 #define VX855_GPIO(n) (NR_VX855_GPI + NR_VX855_GPO + (n))
23 #include "olpc_dcon.h"
25 /* Hardware setup on the XO 1.5:
26 * DCONLOAD connects to
27 * VX855_GPIO1 (not SMBCK2)
28 * DCONBLANK connects to VX855_GPIO8 (not SSPICLK) unused in driver
29 * DCONSTAT0 connects to VX855_GPI10 (not SSPISDI)
30 * DCONSTAT1 connects to VX855_GPI11 (not nSSPISS)
31 * DCONIRQ connects to VX855_GPIO12
32 * DCONSMBDATA connects to VX855 graphics CRTSPD
33 * DCONSMBCLK connects to VX855 graphics CRTSPCLK
36 #define VX855_GENL_PURPOSE_OUTPUT 0x44c // PMIO_Rx4c-4f
37 #define VX855_GPI_STATUS_CHG 0x450 // PMIO_Rx50
38 #define VX855_GPI_SCI_SMI 0x452 // PMIO_Rx52
39 #define BIT_GPIO12 0x40
41 #define PREFIX "OLPC DCON:"
43 static void dcon_clear_irq(void)
45 /* irq status will appear in PMIO_Rx50[6] (RW1C) on gpio12 */
46 outb(BIT_GPIO12, VX855_GPI_STATUS_CHG);
49 static int dcon_was_irq(void)
53 /* irq status will appear in PMIO_Rx50[6] on gpio12 */
54 tmp = inb(VX855_GPI_STATUS_CHG);
55 return !!(tmp & BIT_GPIO12);
60 static int dcon_init_xo_1_5(struct dcon_priv *dcon)
67 pdev = pci_get_device(PCI_VENDOR_ID_VIA,
68 PCI_DEVICE_ID_VIA_VX855, NULL);
70 printk(KERN_ERR "cannot find VX855 PCI ID\n");
74 pci_read_config_byte(pdev, 0x95, &tmp);
75 pci_write_config_byte(pdev, 0x95, tmp|0x0c);
77 /* Set GPIO8 to GPIO mode, not SSPICLK */
78 pci_read_config_byte(pdev, 0xe3, &tmp);
79 pci_write_config_byte(pdev, 0xe3, tmp | 0x04);
81 /* Set GPI10/GPI11 to GPI mode, not SSPISDI/SSPISS */
82 pci_read_config_byte(pdev, 0xe4, &tmp);
83 pci_write_config_byte(pdev, 0xe4, tmp|0x08);
85 /* clear PMU_RxE1[6] to select SCI on GPIO12 */
86 /* clear PMU_RxE0[6] to choose falling edge */
87 pci_read_config_byte(pdev, 0xe1, &tmp);
88 pci_write_config_byte(pdev, 0xe1, tmp & ~BIT_GPIO12);
89 pci_read_config_byte(pdev, 0xe0, &tmp);
90 pci_write_config_byte(pdev, 0xe0, tmp & ~BIT_GPIO12);
94 /* set PMIO_Rx52[6] to enable SCI/SMI on gpio12 */
95 outb(inb(VX855_GPI_SCI_SMI)|BIT_GPIO12, VX855_GPI_SCI_SMI);
97 /* Determine the current state of DCONLOAD, likely set by firmware */
99 dcon->curr_src = (inl(VX855_GENL_PURPOSE_OUTPUT) & 0x1000) ?
100 DCON_SOURCE_CPU : DCON_SOURCE_DCON;
101 dcon->pending_src = dcon->curr_src;
105 /* we're sharing the IRQ with ACPI */
106 irq = acpi_gbl_FADT.sci_interrupt;
107 if (request_irq(irq, &dcon_interrupt, IRQF_SHARED, "DCON", dcon)) {
108 printk(KERN_ERR PREFIX "DCON (IRQ%d) allocation failed\n", irq);
115 static void set_i2c_line(int sda, int scl)
118 unsigned int port = 0x26;
120 /* FIXME: This directly accesses the CRT GPIO controller !!! */
141 static void dcon_wiggle_xo_1_5(void)
146 * According to HiMax, when powering the DCON up we should hold
147 * SMB_DATA high for 8 SMB_CLK cycles. This will force the DCON
148 * state machine to reset to a (sane) initial state. Mitch Bradley
149 * did some testing and discovered that holding for 16 SMB_CLK cycles
150 * worked a lot more reliably, so that's what we do here.
154 for (x = 0; x < 16; x++) {
162 /* set PMIO_Rx52[6] to enable SCI/SMI on gpio12 */
163 outb(inb(VX855_GPI_SCI_SMI)|BIT_GPIO12, VX855_GPI_SCI_SMI);
166 static void dcon_set_dconload_xo_1_5(int val)
168 gpio_set_value(VX855_GPIO(1), val);
171 static u8 dcon_read_status_xo_1_5(void)
178 // i believe this is the same as "inb(0x44b) & 3"
179 status = gpio_get_value(VX855_GPI(10));
180 status |= gpio_get_value(VX855_GPI(11)) << 1;
187 struct dcon_platform_data dcon_pdata_xo_1_5 = {
188 .init = dcon_init_xo_1_5,
189 .bus_stabilize_wiggle = dcon_wiggle_xo_1_5,
190 .set_dconload = dcon_set_dconload_xo_1_5,
191 .read_status = dcon_read_status_xo_1_5,