blob: 69415eec425c8a3c18876f8304a87d0f3a115567 [file] [log] [blame]
Andres Salomoneecb3e42010-09-24 19:13:42 -07001/*
2 * Copyright (c) 2009,2010 One Laptop per Child
3 *
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.
7 */
8
9#include <linux/acpi.h>
Sascha Silbebed4ab72011-04-03 12:05:36 +020010#include <linux/delay.h>
Andres Salomon097cd832011-02-10 17:53:24 -080011#include <linux/pci.h>
12#include <linux/gpio.h>
13#include <asm/olpc.h>
14
15/* TODO: this eventually belongs in linux/vx855.h */
16#define NR_VX855_GPI 14
17#define NR_VX855_GPO 13
18#define NR_VX855_GPIO 15
19
20#define VX855_GPI(n) (n)
21#define VX855_GPO(n) (NR_VX855_GPI + (n))
22#define VX855_GPIO(n) (NR_VX855_GPI + NR_VX855_GPO + (n))
23
24#include "olpc_dcon.h"
Andres Salomoneecb3e42010-09-24 19:13:42 -070025
26/* Hardware setup on the XO 1.5:
Stefan Brählera3712f42011-04-12 19:53:31 +020027 * DCONLOAD connects to VX855_GPIO1 (not SMBCK2)
28 * DCONBLANK connects to VX855_GPIO8 (not SSPICLK) unused in driver
Andres Salomoneecb3e42010-09-24 19:13:42 -070029 * DCONSTAT0 connects to VX855_GPI10 (not SSPISDI)
30 * DCONSTAT1 connects to VX855_GPI11 (not nSSPISS)
Andres Salomon8f2fb162011-02-10 17:49:58 -080031 * DCONIRQ connects to VX855_GPIO12
Andres Salomoneecb3e42010-09-24 19:13:42 -070032 * DCONSMBDATA connects to VX855 graphics CRTSPD
33 * DCONSMBCLK connects to VX855 graphics CRTSPCLK
34 */
35
Stefan Brählera3712f42011-04-12 19:53:31 +020036#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 */
Andres Salomoneecb3e42010-09-24 19:13:42 -070039#define BIT_GPIO12 0x40
40
41#define PREFIX "OLPC DCON:"
42
Andres Salomoneecb3e42010-09-24 19:13:42 -070043static void dcon_clear_irq(void)
44{
Andres Salomon8f2fb162011-02-10 17:49:58 -080045 /* irq status will appear in PMIO_Rx50[6] (RW1C) on gpio12 */
46 outb(BIT_GPIO12, VX855_GPI_STATUS_CHG);
Andres Salomoneecb3e42010-09-24 19:13:42 -070047}
48
49static int dcon_was_irq(void)
50{
51 u_int8_t tmp;
52
Andres Salomon8f2fb162011-02-10 17:49:58 -080053 /* irq status will appear in PMIO_Rx50[6] on gpio12 */
54 tmp = inb(VX855_GPI_STATUS_CHG);
55 return !!(tmp & BIT_GPIO12);
Andres Salomoneecb3e42010-09-24 19:13:42 -070056
57 return 0;
58}
59
Andres Salomonbbe963f2011-02-10 17:54:24 -080060static int dcon_init_xo_1_5(struct dcon_priv *dcon)
Andres Salomoneecb3e42010-09-24 19:13:42 -070061{
62 unsigned int irq;
63 u_int8_t tmp;
64 struct pci_dev *pdev;
Stefan Brählera3712f42011-04-12 19:53:31 +020065
Andres Salomoneecb3e42010-09-24 19:13:42 -070066 pdev = pci_get_device(PCI_VENDOR_ID_VIA,
67 PCI_DEVICE_ID_VIA_VX855, NULL);
68 if (!pdev) {
69 printk(KERN_ERR "cannot find VX855 PCI ID\n");
70 return 1;
71 }
72
Andres Salomon8f2fb162011-02-10 17:49:58 -080073 pci_read_config_byte(pdev, 0x95, &tmp);
74 pci_write_config_byte(pdev, 0x95, tmp|0x0c);
Andres Salomoneecb3e42010-09-24 19:13:42 -070075
76 /* Set GPIO8 to GPIO mode, not SSPICLK */
77 pci_read_config_byte(pdev, 0xe3, &tmp);
78 pci_write_config_byte(pdev, 0xe3, tmp | 0x04);
79
80 /* Set GPI10/GPI11 to GPI mode, not SSPISDI/SSPISS */
81 pci_read_config_byte(pdev, 0xe4, &tmp);
82 pci_write_config_byte(pdev, 0xe4, tmp|0x08);
83
Andres Salomon8f2fb162011-02-10 17:49:58 -080084 /* clear PMU_RxE1[6] to select SCI on GPIO12 */
85 /* clear PMU_RxE0[6] to choose falling edge */
86 pci_read_config_byte(pdev, 0xe1, &tmp);
87 pci_write_config_byte(pdev, 0xe1, tmp & ~BIT_GPIO12);
88 pci_read_config_byte(pdev, 0xe0, &tmp);
89 pci_write_config_byte(pdev, 0xe0, tmp & ~BIT_GPIO12);
Andres Salomoneecb3e42010-09-24 19:13:42 -070090
Andres Salomon8f2fb162011-02-10 17:49:58 -080091 dcon_clear_irq();
Andres Salomoneecb3e42010-09-24 19:13:42 -070092
Andres Salomon8f2fb162011-02-10 17:49:58 -080093 /* set PMIO_Rx52[6] to enable SCI/SMI on gpio12 */
94 outb(inb(VX855_GPI_SCI_SMI)|BIT_GPIO12, VX855_GPI_SCI_SMI);
Andres Salomoneecb3e42010-09-24 19:13:42 -070095
96 /* Determine the current state of DCONLOAD, likely set by firmware */
Andres Salomon8f2fb162011-02-10 17:49:58 -080097 /* GPIO1 */
Andres Salomonbbe963f2011-02-10 17:54:24 -080098 dcon->curr_src = (inl(VX855_GENL_PURPOSE_OUTPUT) & 0x1000) ?
Andres Salomoneecb3e42010-09-24 19:13:42 -070099 DCON_SOURCE_CPU : DCON_SOURCE_DCON;
Andres Salomonbbe963f2011-02-10 17:54:24 -0800100 dcon->pending_src = dcon->curr_src;
Andres Salomoneecb3e42010-09-24 19:13:42 -0700101
102 pci_dev_put(pdev);
103
104 /* we're sharing the IRQ with ACPI */
105 irq = acpi_gbl_FADT.sci_interrupt;
Andres Salomonbbe963f2011-02-10 17:54:24 -0800106 if (request_irq(irq, &dcon_interrupt, IRQF_SHARED, "DCON", dcon)) {
Andres Salomoneecb3e42010-09-24 19:13:42 -0700107 printk(KERN_ERR PREFIX "DCON (IRQ%d) allocation failed\n", irq);
108 return 1;
109 }
110
Andres Salomoneecb3e42010-09-24 19:13:42 -0700111 return 0;
112}
113
114static void set_i2c_line(int sda, int scl)
115{
116 unsigned char tmp;
117 unsigned int port = 0x26;
118
119 /* FIXME: This directly accesses the CRT GPIO controller !!! */
120 outb(port, 0x3c4);
121 tmp = inb(0x3c5);
122
123 if (scl)
124 tmp |= 0x20;
125 else
126 tmp &= ~0x20;
127
128 if (sda)
129 tmp |= 0x10;
130 else
131 tmp &= ~0x10;
132
133 tmp |= 0x01;
134
135 outb(port, 0x3c4);
136 outb(tmp, 0x3c5);
137}
138
139
140static void dcon_wiggle_xo_1_5(void)
141{
142 int x;
143
144 /*
145 * According to HiMax, when powering the DCON up we should hold
146 * SMB_DATA high for 8 SMB_CLK cycles. This will force the DCON
147 * state machine to reset to a (sane) initial state. Mitch Bradley
148 * did some testing and discovered that holding for 16 SMB_CLK cycles
149 * worked a lot more reliably, so that's what we do here.
Stefan Brählera3712f42011-04-12 19:53:31 +0200150 */
Andres Salomoneecb3e42010-09-24 19:13:42 -0700151 set_i2c_line(1, 1);
152
153 for (x = 0; x < 16; x++) {
154 udelay(5);
155 set_i2c_line(1, 0);
156 udelay(5);
157 set_i2c_line(1, 1);
158 }
159 udelay(5);
160
Andres Salomon8f2fb162011-02-10 17:49:58 -0800161 /* set PMIO_Rx52[6] to enable SCI/SMI on gpio12 */
162 outb(inb(VX855_GPI_SCI_SMI)|BIT_GPIO12, VX855_GPI_SCI_SMI);
Andres Salomoneecb3e42010-09-24 19:13:42 -0700163}
164
165static void dcon_set_dconload_xo_1_5(int val)
166{
Andres Salomon8f2fb162011-02-10 17:49:58 -0800167 gpio_set_value(VX855_GPIO(1), val);
Andres Salomoneecb3e42010-09-24 19:13:42 -0700168}
169
Xi Wang91762052011-12-02 16:28:43 -0500170static int dcon_read_status_xo_1_5(u8 *status)
Andres Salomoneecb3e42010-09-24 19:13:42 -0700171{
Andres Salomoneecb3e42010-09-24 19:13:42 -0700172 if (!dcon_was_irq())
173 return -1;
174
Stefan Brählera3712f42011-04-12 19:53:31 +0200175 /* i believe this is the same as "inb(0x44b) & 3" */
Xi Wang91762052011-12-02 16:28:43 -0500176 *status = gpio_get_value(VX855_GPI(10));
177 *status |= gpio_get_value(VX855_GPI(11)) << 1;
Andres Salomoneecb3e42010-09-24 19:13:42 -0700178
179 dcon_clear_irq();
180
Xi Wang91762052011-12-02 16:28:43 -0500181 return 0;
Andres Salomoneecb3e42010-09-24 19:13:42 -0700182}
183
Andres Salomon097cd832011-02-10 17:53:24 -0800184struct dcon_platform_data dcon_pdata_xo_1_5 = {
Andres Salomoneecb3e42010-09-24 19:13:42 -0700185 .init = dcon_init_xo_1_5,
186 .bus_stabilize_wiggle = dcon_wiggle_xo_1_5,
187 .set_dconload = dcon_set_dconload_xo_1_5,
188 .read_status = dcon_read_status_xo_1_5,
189};