blob: b54e10043a1e28afc78f03b5bf7d3d8039eaa0bd [file] [log] [blame]
Angel Pons7544e2f2020-04-03 01:23:10 +02001/* SPDX-License-Identifier: GPL-2.0-only */
Stefan Reinauer838c5a52010-01-17 14:08:17 +00002
3#include <types.h>
Stefan Reinauer838c5a52010-01-17 14:08:17 +00004#include <console/console.h>
5#include <arch/io.h>
6#include <delay.h>
7
Sven Schnelle7592e8b2011-01-27 11:43:03 +00008#include <ec/acpi/ec.h>
Stefan Reinauer838c5a52010-01-17 14:08:17 +00009#include "m3885.h"
10
11#define TH0LOW 80
12#define TH0HIGH 85
13#define TH0CRIT 120
14#define TH1LOW 75
15#define TH1HIGH 80
16
17static u8 variables[] = {
18 /* Offs, AND, OR */
Elyes HAOUASec16e932016-10-07 18:22:44 +020019 0x08, 0x48, 0x6C, /* Keyboard ScanCode Set & LED Data (kState1) */
20 0x0a, 0x01, 0x00, /* Keyboard Shift flags (kState3) */
21 0x0c, 0x80, 0x08, /* Keyboard State flags (kState5) */
22 0x11, 0xff, 0x06, /* Make/Break Debounce #'s (debounce) */
23 0x13, 0xff, 0x00, /* HotKey1 ScanCode (hotKey1) */
24 0x14, 0xff, 0x00, /* HotKey2 ScanCode (hotKey2) */
25 0x15, 0xff, 0x3f, /* HotKey3 ScanCode (hotKey3) */
26 0x16, 0xff, 0x00, /* HotKey4 ScanCode (hotKey4) */
27 0x17, 0xff, 0x00, /* HotKey5 ScanCode (hotKey5) */
28 0x18, 0xff, 0x0e, /* HotKey6 ScanCode (hotKey6) */
29 0x19, 0xff, 0x9f, /* HotKey1 Task = c5 Command Data (keyTsk1) */
30 0x1a, 0xff, 0x9f, /* HotKey2 Task = c5 Command Data (keyTsk2) */
31 0x1b, 0xff, 0x6a, /* HotKey3 Task = c5 Command Data (keyTsk3) */
32 0x1c, 0xff, 0x9f, /* HotKey4 Task = c5 Command Data (keyTsk4) */
33 0x1d, 0xff, 0x9f, /* HotKey5 Task = c5 Command Data (keyTsk5) */
34 0x1e, 0xff, 0x87, /* FuncKey Task = c5 Command Data (funcTsk) */
35 0x1f, 0xff, 0x9f, /* Delayed Task = c5 Command Data (dlyTsk1) */
36 0x20, 0xff, 0x9f, /* Wake-Up Task = c5 Command Data (wakeTsk) */
37
38 0x21, 0xff, 0x08, /* WigglePin Pulse Width * 2.4ms (tmPulse) */
39 0x24, 0xff, 0x30, /* Keyboard State Flags (kState7) */
40
41 0x2b, 0xff, 0x00,
42 0x2c, 0xff, 0x80, /* Set Fn-Key 8 */
43 0x2d, 0xff, 0x02, /* Set Fn-Key 9 */
44 0x2e, 0xff, 0x00, /* Set Fn-Key 1-8 task (0 = SMI) */
45 0x2f, 0xff, 0x00, /* Set Fn-Key 9-12 task (1 = SCI) */
Stefan Reinauer838c5a52010-01-17 14:08:17 +000046};
47
48static u8 matrix[] = {
Elyes HAOUASec16e932016-10-07 18:22:44 +020049 0xc1,0xc0,0xd8,0xdb,0xbf,0x05,0x76,0xbf, /* (0x00-0x07) */
50 0xbf,0x80,0x78,0xbf,0xbf,0x07,0x88,0xc2, /* (0x08-0x0f) */
51 0x03,0x09,0xd9,0x16,0xbf,0x06,0x0e,0x81, /* (0x10-0x17) */
52 0xbf,0xbf,0xee,0xbf,0xbf,0x55,0x9a,0x89, /* (0x18-0x1f) */
53 0x1e,0x15,0x36,0xda,0xe8,0xbf,0x0d,0xbf, /* (0x20-0x27) */
54 0xbf,0xbf,0xbf,0xa3,0xbf,0x4e,0x66,0x8b, /* (0x28-0x2f) */
55 0x1d,0x2e,0xe6,0xe7,0xe5,0x1c,0x58,0xbf, /* (0x30-0x37) */
56 0x82,0xbf,0xf0,0xbf,0xbf,0x5b,0x5d,0x8c, /* (0x38-0x3f) */
57 0x22,0x25,0x2c,0x35,0xe1,0x1a,0x96,0xbf, /* (0x40-0x47) */
58 0xbf,0xbf,0xec,0xbf,0xbf,0x54,0xf1,0x8f, /* (0x48-0x4f) */
59 0x1b,0x2a,0x2b,0x32,0xe9,0x31,0x29,0x61, /* (0x50-0x57) */
60 0xbf,0xbf,0x8d,0xbf,0x86,0xc3,0x92,0x93, /* (0x58-0x5f) */
61 0x21,0x23,0x34,0x33,0x41,0xe0,0xbf,0xbf, /* (0x60-0x67) */
62 0xbf,0x85,0xeb,0xbf,0xb6,0xbf,0x91,0xbf, /* (0x68-0x6f) */
63 0x26,0x24,0x2d,0xe3,0xe2,0xe4,0xbf,0xbf, /* (0x70-0x77) */
64 0x87,0xbf,0xea,0xbf,0xbf,0x52,0x90,0x8e, /* (0x78-0x7f) */
Stefan Reinauer838c5a52010-01-17 14:08:17 +000065};
66
67static u8 function_ram[] = {
Elyes HAOUASec16e932016-10-07 18:22:44 +020068 0x04,0xbd,0x0c,0xbe,0x7e,0x9a,0x8a,0xb6, /* (0xc0-0xc3) */
69 0x92,0x8f,0x93,0x8e,0x81,0x86,0x82,0x87, /* (0xc4-0xc7) */
70 0x8a,0x9a,0x8d,0x7e,0x88,0x84,0x7e,0x78, /* (0xc8-0xcb) */
71 0x77,0x07,0x77,0x98,0x89,0xb2,0x05,0x9b, /* (0xcc-0xcf) */
72 0x78,0x84,0x07,0x88,0x8a,0x7e,0x05,0xa6, /* (0xd0-0xd3) */
73 0x06,0xa7,0x04,0xa8,0x0c,0xa9,0x03,0xaa, /* (0xd4-0xd7) */
74 0x0b,0xc1,0x83,0xc0,0x0a,0xad,0x01,0xae, /* (0xd8-0xdb) */
75 0x09,0xaf,0x78,0xb0,0x07,0xb1,0x1a,0x61, /* (0xdc-0xdf) */
76 0x3b,0x69,0x42,0x72,0x4b,0x7a,0x3c,0x6b, /* (0xe0-0xe3) */
77 0x43,0x73,0x44,0x74,0x3d,0x6c,0x3e,0x75, /* (0xe4-0xe7) */
78 0x46,0x7d,0x3a,0x70,0x49,0x71,0x4a,0x94, /* (0xe8-0xeb) */
79 0x4c,0x79,0x4c,0x7c,0x45,0x7c,0x45,0x79, /* (0xec-0xef) */
80 0x4d,0x7b,0x5a,0x95,0x4c,0x7b,0x45,0x7b, /* (0xf0-0xf3) */
81 0x4d,0x79,0x4d,0x7c,0x4e,0x7b,0x54,0x95, /* (0xf4-0xf7) */
82 0x52,0x7c,0x45,0x94,0x4a,0x79,0xb3,0x95, /* (0xf8-0xfb) */
83 0xb4,0x7b,0xb5,0x7c,0x00,0x00,0x55,0x79, /* (0xfc-0xff) */
Stefan Reinauer838c5a52010-01-17 14:08:17 +000084};
85
Stefan Reinauer838c5a52010-01-17 14:08:17 +000086#define KBD_DATA 0x60
87#define KBD_SC 0x64
88
Elyes HAOUASec16e932016-10-07 18:22:44 +020089#define KBD_IBF (1 << 1) /* 1: input buffer full (data ready for ec) */
90#define KBD_OBF (1 << 0) /* 1: output buffer full (data ready for host) */
Stefan Reinauer838c5a52010-01-17 14:08:17 +000091
92static int send_kbd_command(u8 command)
93{
94 int timeout;
95
96 timeout = 0x7ff;
97 while ((inb(KBD_SC) & KBD_IBF) && --timeout) {
98 udelay(10);
99 if ((timeout & 0xff) == 0)
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000100 printk(BIOS_SPEW, ".");
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000101 }
102 if (!timeout) {
Stefan Reinauer14e22772010-04-27 06:56:47 +0000103 printk(BIOS_DEBUG, "Timeout while sending command 0x%02x to EC!\n",
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000104 command);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000105 }
106
107 outb(command, KBD_SC);
108 return 0;
109}
110
111static int send_kbd_data(u8 data)
112{
113 int timeout;
114
115 timeout = 0x7ff;
Elyes HAOUASec16e932016-10-07 18:22:44 +0200116 while ((inb(KBD_SC) & KBD_IBF) && --timeout) { /* wait for IBF = 0 */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000117 udelay(10);
118 if ((timeout & 0xff) == 0)
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000119 printk(BIOS_SPEW, ".");
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000120 }
121 if (!timeout) {
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000122 printk(BIOS_DEBUG, "Timeout while sending data 0x%02x to EC!\n",
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000123 data);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000124 }
125
126 outb(data, KBD_DATA);
127
128 return 0;
129}
130
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000131static u8 recv_kbd_data(void)
132{
133 int timeout;
134 u8 data;
135
136 timeout = 0x7fff;
Elyes HAOUASec16e932016-10-07 18:22:44 +0200137 while (--timeout) { /* Wait for OBF = 1 */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000138 if (inb(KBD_SC) & KBD_OBF) {
139 break;
140 }
141 udelay(10);
142 if ((timeout & 0xff) == 0)
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000143 printk(BIOS_SPEW, ".");
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000144 }
145 if (!timeout) {
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000146 printk(BIOS_DEBUG, "\nTimeout while receiving data from EC!\n");
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000147 }
148
149 data = inb(KBD_DATA);
150
151 return data;
152}
153
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000154static u8 m3885_get_variable(u8 index)
155{
156 u8 ret;
157
158 send_kbd_command(0xb8);
159 send_kbd_data(index);
160 send_kbd_command(0xbc);
161 send_kbd_command(0xff);
162 ret = recv_kbd_data();
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000163 printk(BIOS_SPEW, "m3885: get variable %02x = %02x\n", index, ret);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000164 return ret;
165}
166
167static void m3885_set_variable(u8 index, u8 data)
168{
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000169 printk(BIOS_SPEW, "m3885: set variable %02x = %02x\n", index, data);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000170 send_kbd_command(0xb8);
171 send_kbd_data(index);
172 send_kbd_command(0xbd);
173 send_kbd_data(data);
174}
175
176static void m3885_set_proc_ram(u8 index, u8 data)
177{
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000178 printk(BIOS_SPEW, "m3885: set procram %02x = %02x\n", index, data);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000179 send_kbd_command(0xb8);
180 send_kbd_data(index);
181 send_kbd_command(0xbb);
182 send_kbd_data(data);
183}
184
185static u8 m3885_get_proc_ram(u8 index)
186{
187 u8 ret;
188
189 send_kbd_command(0xb8);
190 send_kbd_data(index);
191 send_kbd_command(0xba);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000192 ret = recv_kbd_data();
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000193 printk(BIOS_SPEW, "m3885: get procram %02x = %02x\n", index, ret);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000194 return ret;
195}
196
197static u8 m3885_read_port(void)
198{
199 u8 reg8;
200
201 reg8 = m3885_get_variable(0x0c);
202 reg8 &= ~(7 << 4);
Elyes HAOUASec16e932016-10-07 18:22:44 +0200203 reg8 |= (4 << 4); /* bank 4 */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000204 m3885_set_variable(0x0c, reg8);
205
206 /* P6YSTATE */
207 return m3885_get_proc_ram(0xf8);
208}
209
210void m3885_configure_multikey(void)
211{
212 int i;
213 u8 reg8;
214 u8 kstate5_flags, offs, maxvars;
215
Elyes HAOUAS8ab989e2016-07-30 17:46:17 +0200216 /* RAM bank 0 */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000217 kstate5_flags = m3885_get_variable(0x0c);
218 m3885_set_variable(0x0c, kstate5_flags & ~(7 << 4));
219
220 /* Write Matrix to bank 0 */
Elyes HAOUASa5aad2e2016-09-19 09:47:16 -0600221 for (i = 0; i < ARRAY_SIZE(matrix); i++) {
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000222 m3885_set_proc_ram(i + 0x80, matrix[i]);
223 }
224
Stefan Reinauer14e22772010-04-27 06:56:47 +0000225
Elyes HAOUAS8ab989e2016-07-30 17:46:17 +0200226 /* RAM bank 2 */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000227 m3885_set_variable(0x0c, (kstate5_flags & (~(7 << 4))) | (2 << 4));
228
229 /* Get the number of variables */
230 maxvars = m3885_get_variable(0x00);
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000231 printk(BIOS_DEBUG, "M388x has %d variables in bank 2.\n", maxvars);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000232 if (maxvars >= 35) {
233 offs = m3885_get_variable(0x23);
234 if ((offs > 0xc0) || (offs < 0x80)) {
Elyes HAOUAS8ab989e2016-07-30 17:46:17 +0200235 printk(BIOS_DEBUG, "M388x does not have a valid RAM offset (0x%x)\n", offs);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000236 } else {
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000237 printk(BIOS_DEBUG, "Writing Fn-Table to M388x RAM offset 0x%x\n", offs);
Elyes HAOUASa5aad2e2016-09-19 09:47:16 -0600238 for (i = 0; i < ARRAY_SIZE(function_ram); i++) {
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000239 m3885_set_proc_ram(i + offs, function_ram[i]);
240 }
241 }
242 } else {
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000243 printk(BIOS_DEBUG, "Could not load Function-RAM (%d).\n", maxvars);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000244 }
245
Elyes HAOUASec16e932016-10-07 18:22:44 +0200246 /* restore original bank */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000247 m3885_set_variable(0x0c, kstate5_flags);
248 maxvars = m3885_get_variable(0x00);
Stefan Reinauerc02b4fc2010-03-22 11:42:32 +0000249 printk(BIOS_DEBUG, "M388x has %d variables in original bank.\n", maxvars);
Elyes HAOUASa5aad2e2016-09-19 09:47:16 -0600250 for (i = 0; i < ARRAY_SIZE(variables); i+=3) {
Elyes HAOUAS5ba154a2020-08-04 13:27:52 +0200251 if (variables[i + 0] > maxvars)
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000252 continue;
253 reg8 = m3885_get_variable(variables[i + 0]);
254 reg8 &= ~(variables[i + 1]);
Elyes HAOUASec16e932016-10-07 18:22:44 +0200255 reg8 |= variables[i + 2];
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000256 m3885_set_variable(variables[i + 0], reg8);
257 }
258
259 /* OEM Init */
260
261 /* Set Bank 1 */
262 m3885_set_variable(0x0c, (kstate5_flags & ~(7 << 4)) | (1 << 4));
263
264 /* Set SMI# at P5.1 */
265 /* SMI Control -> p5.1 = EXTSMI# */
266 m3885_set_proc_ram(0xff, 0xc1);
267
268 /* Set Fn-Key Task 0 -> SMI#/SCI */
269 m3885_set_proc_ram(0x6d, 0x81);
270
271 /* Set Fn-Key Task 1 -> SCI */
272 m3885_set_proc_ram(0x6c, 0x80);
273
274 /* Number of Thermal Sensors */
275 m3885_set_proc_ram(0xf2, 0x02);
276
277 /* Critical Task */
278 m3885_set_proc_ram(0xf3, 0x5d);
Stefan Reinauer14e22772010-04-27 06:56:47 +0000279
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000280 /* Thermal Polling Period */
281 m3885_set_proc_ram(0xf9, 0x0a);
282
283 /* ReadPort */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000284
285 /* AC PRESN# */
286 if (m3885_read_port() & (1 << 0))
287 reg8 = 0x8a;
288 else
289 reg8 = 0x9a;
Elyes HAOUASec16e932016-10-07 18:22:44 +0200290 m3885_set_proc_ram(0xd0, reg8); /* P60SPEC */
Stefan Reinauer14e22772010-04-27 06:56:47 +0000291
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000292 /* SENSE1# */
293 if (m3885_read_port() & (1 << 2))
294 reg8 = 0x8a;
295 else
296 reg8 = 0x9a;
Elyes HAOUASec16e932016-10-07 18:22:44 +0200297 m3885_set_proc_ram(0xd2, reg8); /* P62SPEC */
Stefan Reinauer14e22772010-04-27 06:56:47 +0000298
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000299 /* SENSE2# */
300 if (m3885_read_port() & (1 << 3))
301 reg8 = 0x8a;
302 else
303 reg8 = 0x9a;
Elyes HAOUASec16e932016-10-07 18:22:44 +0200304 m3885_set_proc_ram(0xd3, reg8); /* P63SPEC */
Stefan Reinauer14e22772010-04-27 06:56:47 +0000305
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000306 /* Low Active Port */
Elyes HAOUASec16e932016-10-07 18:22:44 +0200307 m3885_set_proc_ram(0xd1, 0x88); /* P61SPEC */
308 m3885_set_proc_ram(0xd6, 0x88); /* P66SPEC */
309 m3885_set_proc_ram(0xd7, 0x88); /* P67SPEC */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000310
311 /* High Active Port */
Elyes HAOUASec16e932016-10-07 18:22:44 +0200312 m3885_set_proc_ram(0xd4, 0x98); /* P64SPEC */
313 m3885_set_proc_ram(0xd5, 0x98); /* P65SPEC */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000314
315 /* Set P60TASK-P67TASK */
316 /* SCI */
Elyes HAOUASec16e932016-10-07 18:22:44 +0200317 m3885_set_proc_ram(0xda, 0x80); /* P62TASK SENSE1# */
318 m3885_set_proc_ram(0xdb, 0x80); /* P63TASK SENSE2# */
319 m3885_set_proc_ram(0xdd, 0x80); /* P65TASK PROCHOT */
320 m3885_set_proc_ram(0xde, 0x80); /* P65TASK THERMTRIP# */
321 m3885_set_proc_ram(0xdf, 0x80); /* P65TASK PME# */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000322 /* SMI/SCI */
Elyes HAOUASec16e932016-10-07 18:22:44 +0200323 m3885_set_proc_ram(0xd8, 0x81); /* P60TASK, AC_PRESN# */
324 m3885_set_proc_ram(0xd9, 0x81); /* P61TASK, LID# */
325 m3885_set_proc_ram(0xdc, 0x81); /* P64TASK, FDD/LPT# */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000326
327 /* Thermal */
328 /* Bank 5 */
329 m3885_set_variable(0x0c, (kstate5_flags & ~(7 << 4)) | (5 << 4));
330
331 /* Thermal 0: Active cooling, Speed Step Down */
Elyes HAOUASec16e932016-10-07 18:22:44 +0200332 m3885_set_proc_ram(0x81, 0x9c); /* THRM0 */
333 m3885_set_proc_ram(0x82, 0x01); /* THRM0 CMD */
334 m3885_set_proc_ram(0x84, TH0LOW); /* THRM0 LOW */
335 m3885_set_proc_ram(0x85, TH0HIGH); /* THRM0 HIGH */
336 m3885_set_proc_ram(0x86, 0x81); /* Set Task SMI#/SCI */
337 m3885_set_proc_ram(0x87, TH0CRIT); /* THRM0 CRIT */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000338
339 /* Thermal 1: Passive cooling, Fan On */
Elyes HAOUASec16e932016-10-07 18:22:44 +0200340 m3885_set_proc_ram(0x89, 0x9c); /* THRM1 */
341 m3885_set_proc_ram(0x8a, 0x01); /* THRM1 CMD */
342 m3885_set_proc_ram(0x8c, TH1LOW); /* THRM1 LOW */
343 m3885_set_proc_ram(0x8d, TH1HIGH); /* THRM1 HIGH */
344 m3885_set_proc_ram(0x8e, 0x81); /* Set Task SMI#/SCI */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000345
346 /* Switch Task to SMI */
Elyes HAOUASec16e932016-10-07 18:22:44 +0200347 udelay(100 * 1000); /* 100ms */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000348 outb(KBD_SC, 0xca);
Elyes HAOUASec16e932016-10-07 18:22:44 +0200349 udelay(100 * 1000); /* 100ms */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000350 outb(KBD_DATA, 0x17);
351
352 /* Set P22 to high level, keyboard backlight default off */
Elyes HAOUASec16e932016-10-07 18:22:44 +0200353 udelay(100 * 1000); /* 100ms */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000354 outb(KBD_SC, 0xc5);
Elyes HAOUASec16e932016-10-07 18:22:44 +0200355 udelay(100 * 1000); /* 100ms */
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000356 outb(KBD_DATA, 0x4a);
357}
358
359u8 m3885_gpio(u8 value)
360{
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000361 /* First write data */
362 ec_write(M3885_CMDAT1, value);
363
364 /* Issue command: ACCESS GPIO */
365 ec_write(M3885_CMCMD, 0xc5);
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000366 return 0;
Stefan Reinauer838c5a52010-01-17 14:08:17 +0000367}