Blame view

drivers/macintosh/via-macii.c 14.1 KB
b24413180   Greg Kroah-Hartman   License cleanup: ...
1
  // SPDX-License-Identifier: GPL-2.0
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
2
3
4
5
6
7
8
9
10
11
12
13
14
  /*
   * Device driver for the via ADB on (many) Mac II-class machines
   *
   * Based on the original ADB keyboard handler Copyright (c) 1997 Alan Cox
   * Also derived from code Copyright (C) 1996 Paul Mackerras.
   *
   * With various updates provided over the years by Michael Schmitz,
   * Guideo Koerber and others.
   *
   * Rewrite for Unified ADB by Joshua M. Thompson (funaho@jurai.org)
   *
   * 1999-08-02 (jmt) - Initial rewrite for Unified ADB.
   * 2000-03-29 Tony Mantler <tonym@mac.linux-m68k.org>
47fd20606   Finn Thain   macintosh/via-mac...
15
   *            - Big overhaul, should actually work now.
d0c2c269a   Finn Thain   drivers: Clean up...
16
   * 2006-12-31 Finn Thain - Another overhaul.
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
17
18
19
20
21
22
23
24
   *
   * Suggested reading:
   *   Inside Macintosh, ch. 5 ADB Manager
   *   Guide to the Macinstosh Family Hardware, ch. 8 Apple Desktop Bus
   *   Rockwell R6522 VIA datasheet
   *
   * Apple's "ADB Analyzer" bus sniffer is invaluable:
   *   ftp://ftp.apple.com/developer/Tool_Chest/Devices_-_Hardware/Apple_Desktop_Bus/
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
25
   */
47fd20606   Finn Thain   macintosh/via-mac...
26

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
27
28
29
30
31
  #include <stdarg.h>
  #include <linux/types.h>
  #include <linux/errno.h>
  #include <linux/kernel.h>
  #include <linux/delay.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
32
33
34
35
36
  #include <linux/adb.h>
  #include <linux/interrupt.h>
  #include <linux/init.h>
  #include <asm/macintosh.h>
  #include <asm/macints.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
37
  #include <asm/mac_via.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
  
  static volatile unsigned char *via;
  
  /* VIA registers - spaced 0x200 bytes apart */
  #define RS		0x200		/* skip between registers */
  #define B		0		/* B-side data */
  #define A		RS		/* A-side data */
  #define DIRB		(2*RS)		/* B-side direction (1=output) */
  #define DIRA		(3*RS)		/* A-side direction (1=output) */
  #define T1CL		(4*RS)		/* Timer 1 ctr/latch (low 8 bits) */
  #define T1CH		(5*RS)		/* Timer 1 counter (high 8 bits) */
  #define T1LL		(6*RS)		/* Timer 1 latch (low 8 bits) */
  #define T1LH		(7*RS)		/* Timer 1 latch (high 8 bits) */
  #define T2CL		(8*RS)		/* Timer 2 ctr/latch (low 8 bits) */
  #define T2CH		(9*RS)		/* Timer 2 counter (high 8 bits) */
  #define SR		(10*RS)		/* Shift register */
  #define ACR		(11*RS)		/* Auxiliary control register */
  #define PCR		(12*RS)		/* Peripheral control register */
  #define IFR		(13*RS)		/* Interrupt flag register */
  #define IER		(14*RS)		/* Interrupt enable register */
  #define ANH		(15*RS)		/* A-side data, no handshake */
  
  /* Bits in B data register: all active low */
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
61
  #define CTLR_IRQ	0x08		/* Controller rcv status (input) */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
62
63
64
65
66
67
68
69
70
71
72
  #define ST_MASK		0x30		/* mask for selecting ADB state bits */
  
  /* Bits in ACR */
  #define SR_CTRL		0x1c		/* Shift register control bits */
  #define SR_EXT		0x0c		/* Shift on external clock */
  #define SR_OUT		0x10		/* Shift out if 1 */
  
  /* Bits in IFR and IER */
  #define IER_SET		0x80		/* set bits in IER */
  #define IER_CLR		0		/* clear bits in IER */
  #define SR_INT		0x04		/* Shift register full/empty */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
73
74
75
76
77
78
  
  /* ADB transaction states according to GMHW */
  #define ST_CMD		0x00		/* ADB state: command byte */
  #define ST_EVEN		0x10		/* ADB state: even data byte */
  #define ST_ODD		0x20		/* ADB state: odd data byte */
  #define ST_IDLE		0x30		/* ADB state: idle, nothing to send */
f93bfeb55   Finn Thain   macintosh/via-mac...
79
80
81
  /* ADB command byte structure */
  #define ADDR_MASK	0xF0
  #define CMD_MASK	0x0F
b4d76c28e   Finn Thain   macintosh/via-mac...
82
83
  #define OP_MASK		0x0C
  #define TALK		0x0C
f93bfeb55   Finn Thain   macintosh/via-mac...
84

47fd20606   Finn Thain   macintosh/via-mac...
85
  static int macii_init_via(void);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
86
  static void macii_start(void);
7d12e780e   David Howells   IRQ: Maintain reg...
87
  static irqreturn_t macii_interrupt(int irq, void *arg);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
88
89
90
91
92
93
94
95
96
97
98
  static void macii_queue_poll(void);
  
  static int macii_probe(void);
  static int macii_init(void);
  static int macii_send_request(struct adb_request *req, int sync);
  static int macii_write(struct adb_request *req);
  static int macii_autopoll(int devs);
  static void macii_poll(void);
  static int macii_reset_bus(void);
  
  struct adb_driver via_macii_driver = {
3a52f6f98   Finn Thain   macintosh/adb: Us...
99
100
101
102
103
104
105
  	.name         = "Mac II",
  	.probe        = macii_probe,
  	.init         = macii_init,
  	.send_request = macii_send_request,
  	.autopoll     = macii_autopoll,
  	.poll         = macii_poll,
  	.reset_bus    = macii_reset_bus,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
106
107
108
109
110
111
  };
  
  static enum macii_state {
  	idle,
  	sending,
  	reading,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
112
  } macii_state;
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
113
114
115
  static struct adb_request *current_req; /* first request struct in the queue */
  static struct adb_request *last_req;     /* last request struct in the queue */
  static unsigned char reply_buf[16];        /* storage for autopolled replies */
eb4da4cec   Finn Thain   mac68k: macii adb...
116
  static unsigned char *reply_ptr;     /* next byte in reply_buf or req->reply */
f87a16257   Finn Thain   macintosh/via-mac...
117
  static bool reading_reply;       /* store reply in reply_buf else req->reply */
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
118
119
120
  static int data_index;      /* index of the next byte to send from req->data */
  static int reply_len; /* number of bytes received in reply_buf or req->reply */
  static int status;          /* VIA's ADB status bits captured upon interrupt */
b4d76c28e   Finn Thain   macintosh/via-mac...
121
122
  static bool bus_timeout;                   /* no data was sent by the device */
  static bool srq_asserted;    /* have to poll for the device that asserted it */
f93bfeb55   Finn Thain   macintosh/via-mac...
123
  static u8 last_cmd;              /* the most recent command byte transmitted */
b4d76c28e   Finn Thain   macintosh/via-mac...
124
  static u8 last_talk_cmd;    /* the most recent Talk command byte transmitted */
f93bfeb55   Finn Thain   macintosh/via-mac...
125
  static u8 last_poll_cmd; /* the most recent Talk R0 command byte transmitted */
5c0c15a19   Finn Thain   macintosh/via-mac...
126
  static unsigned int autopoll_devs;  /* bits set are device addresses to poll */
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
127

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
128
129
130
  /* Check for MacII style ADB */
  static int macii_probe(void)
  {
47fd20606   Finn Thain   macintosh/via-mac...
131
132
  	if (macintosh_config->adb_type != MAC_ADB_II)
  		return -ENODEV;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
133
134
  
  	via = via1;
351e5ad32   Finn Thain   macintosh/via-mac...
135
136
  	pr_info("adb: Mac II ADB Driver v1.0 for Unified ADB
  ");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
137
138
139
140
  	return 0;
  }
  
  /* Initialize the driver */
3327e58a0   Finn Thain   macintosh/via-mac...
141
  static int macii_init(void)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
142
143
144
  {
  	unsigned long flags;
  	int err;
47fd20606   Finn Thain   macintosh/via-mac...
145

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
146
  	local_irq_save(flags);
47fd20606   Finn Thain   macintosh/via-mac...
147

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
148
  	err = macii_init_via();
47fd20606   Finn Thain   macintosh/via-mac...
149
150
  	if (err)
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
151

5a2394534   Geert Uytterhoeven   m68k/irq: Remove ...
152
  	err = request_irq(IRQ_MAC_ADB, macii_interrupt, 0, "ADB",
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
153
  			  macii_interrupt);
47fd20606   Finn Thain   macintosh/via-mac...
154
155
  	if (err)
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
156
157
  
  	macii_state = idle;
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
158
  out:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
159
  	local_irq_restore(flags);
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
160
  	return err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
161
  }
47fd20606   Finn Thain   macintosh/via-mac...
162
  /* initialize the hardware */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
163
164
165
  static int macii_init_via(void)
  {
  	unsigned char x;
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
166
167
  	/* We want CTLR_IRQ as input and ST_EVEN | ST_ODD as output lines. */
  	via[DIRB] = (via[DIRB] | ST_EVEN | ST_ODD) & ~CTLR_IRQ;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
168
169
170
  
  	/* Set up state: idle */
  	via[B] |= ST_IDLE;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
171
172
173
174
175
176
177
178
179
  
  	/* Shift register on input */
  	via[ACR] = (via[ACR] & ~SR_CTRL) | SR_EXT;
  
  	/* Wipe any pending data and int */
  	x = via[SR];
  
  	return 0;
  }
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
180
  /* Send an ADB poll (Talk Register 0 command prepended to the request queue) */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
181
182
  static void macii_queue_poll(void)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
183
  	static struct adb_request req;
f93bfeb55   Finn Thain   macintosh/via-mac...
184
185
  	unsigned char poll_command;
  	unsigned int poll_addr;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
186

f93bfeb55   Finn Thain   macintosh/via-mac...
187
188
189
190
191
  	/* This only polls devices in the autopoll list, which assumes that
  	 * unprobed devices never assert SRQ. That could happen if a device was
  	 * plugged in after the adb bus scan. Unplugging it again will resolve
  	 * the problem. This behaviour is similar to MacOS.
  	 */
47fd20606   Finn Thain   macintosh/via-mac...
192
193
  	if (!autopoll_devs)
  		return;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
194

f93bfeb55   Finn Thain   macintosh/via-mac...
195
196
197
198
199
200
201
202
203
204
205
206
207
  	/* The device most recently polled may not be the best device to poll
  	 * right now. Some other device(s) may have signalled SRQ (the active
  	 * device won't do that). Or the autopoll list may have been changed.
  	 * Try polling the next higher address.
  	 */
  	poll_addr = (last_poll_cmd & ADDR_MASK) >> 4;
  	if ((srq_asserted && last_cmd == last_poll_cmd) ||
  	    !(autopoll_devs & (1 << poll_addr))) {
  		unsigned int higher_devs;
  
  		higher_devs = autopoll_devs & -(1 << (poll_addr + 1));
  		poll_addr = ffs(higher_devs ? higher_devs : autopoll_devs) - 1;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
208

f93bfeb55   Finn Thain   macintosh/via-mac...
209
210
211
212
213
214
215
216
217
218
  	/* Send a Talk Register 0 command */
  	poll_command = ADB_READREG(poll_addr, 0);
  
  	/* No need to repeat this Talk command. The transceiver will do that
  	 * as long as it is idle.
  	 */
  	if (poll_command == last_cmd)
  		return;
  
  	adb_request(&req, NULL, ADBREQ_NOSEND, 1, poll_command);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
219

d95fd5fce   Finn Thain   m68k: Mac II ADB ...
220
221
222
223
  	req.sent = 0;
  	req.complete = 0;
  	req.reply_len = 0;
  	req.next = current_req;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
224

f93bfeb55   Finn Thain   macintosh/via-mac...
225
  	if (WARN_ON(current_req)) {
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
226
  		current_req = &req;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
227
  	} else {
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
228
229
  		current_req = &req;
  		last_req = &req;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
230
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
231
232
233
234
235
  }
  
  /* Send an ADB request; if sync, poll out the reply 'till it's done */
  static int macii_send_request(struct adb_request *req, int sync)
  {
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
236
  	int err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
237

d95fd5fce   Finn Thain   m68k: Mac II ADB ...
238
  	err = macii_write(req);
5ce6185c2   Finn Thain   macintosh/via-mac...
239
240
  	if (err)
  		return err;
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
241

5ce6185c2   Finn Thain   macintosh/via-mac...
242
  	if (sync)
5f93d7081   Finn Thain   macintosh/via-mac...
243
  		while (!req->complete)
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
244
  			macii_poll();
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
245

5ce6185c2   Finn Thain   macintosh/via-mac...
246
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
247
  }
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
248
  /* Send an ADB request (append to request queue) */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
249
250
  static int macii_write(struct adb_request *req)
  {
5ce6185c2   Finn Thain   macintosh/via-mac...
251
  	unsigned long flags;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
252
253
254
255
  	if (req->nbytes < 2 || req->data[0] != ADB_PACKET || req->nbytes > 15) {
  		req->complete = 1;
  		return -EINVAL;
  	}
47fd20606   Finn Thain   macintosh/via-mac...
256

a5d361fc2   Al Viro   [PATCH] m68k: NUL...
257
  	req->next = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
258
259
260
  	req->sent = 0;
  	req->complete = 0;
  	req->reply_len = 0;
5ce6185c2   Finn Thain   macintosh/via-mac...
261
  	local_irq_save(flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
262
263
264
265
266
267
  	if (current_req != NULL) {
  		last_req->next = req;
  		last_req = req;
  	} else {
  		current_req = req;
  		last_req = req;
47fd20606   Finn Thain   macintosh/via-mac...
268
269
  		if (macii_state == idle)
  			macii_start();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
270
  	}
5ce6185c2   Finn Thain   macintosh/via-mac...
271
272
  
  	local_irq_restore(flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
273
274
275
276
277
278
  	return 0;
  }
  
  /* Start auto-polling */
  static int macii_autopoll(int devs)
  {
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
279
  	unsigned long flags;
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
280

59ea38f6b   Finn Thain   macintosh/via-mac...
281
  	local_irq_save(flags);
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
282
  	/* bit 1 == device 1, and so on. */
5c0c15a19   Finn Thain   macintosh/via-mac...
283
  	autopoll_devs = (unsigned int)devs & 0xFFFE;
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
284

f93bfeb55   Finn Thain   macintosh/via-mac...
285
286
287
288
  	if (!current_req) {
  		macii_queue_poll();
  		if (current_req && macii_state == idle)
  			macii_start();
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
289
290
291
  	}
  
  	local_irq_restore(flags);
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
292

f93bfeb55   Finn Thain   macintosh/via-mac...
293
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
294
295
296
297
298
  }
  
  /* Prod the chip without interrupts */
  static void macii_poll(void)
  {
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
299
  	macii_interrupt(0, NULL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
300
301
302
303
304
  }
  
  /* Reset the bus */
  static int macii_reset_bus(void)
  {
046ace825   Finn Thain   macintosh/via-mac...
305
  	struct adb_request req;
47fd20606   Finn Thain   macintosh/via-mac...
306

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
307
  	/* Command = 0, Address = ignored */
b52dce873   Finn Thain   macintosh/via-mac...
308
309
  	adb_request(&req, NULL, ADBREQ_NOSEND, 1, ADB_BUSRESET);
  	macii_send_request(&req, 1);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
310

d95fd5fce   Finn Thain   m68k: Mac II ADB ...
311
312
  	/* Don't want any more requests during the Global Reset low time. */
  	udelay(3000);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
313
314
315
316
317
318
  	return 0;
  }
  
  /* Start sending ADB packet */
  static void macii_start(void)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
319
320
321
  	struct adb_request *req;
  
  	req = current_req;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
322

d95fd5fce   Finn Thain   m68k: Mac II ADB ...
323
324
325
  	/* Now send it. Be careful though, that first byte of the request
  	 * is actually ADB_PACKET; the real data begins at index 1!
  	 * And req->nbytes is the number of bytes of real data plus one.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
326
  	 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
327

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
328
329
330
331
332
333
334
335
336
  	/* Output mode */
  	via[ACR] |= SR_OUT;
  	/* Load data */
  	via[SR] = req->data[1];
  	/* set ADB state to 'command' */
  	via[B] = (via[B] & ~ST_MASK) | ST_CMD;
  
  	macii_state = sending;
  	data_index = 2;
b4d76c28e   Finn Thain   macintosh/via-mac...
337
338
339
  
  	bus_timeout = false;
  	srq_asserted = false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
340
341
342
  }
  
  /*
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
343
344
345
346
   * The notorious ADB interrupt handler - does all of the protocol handling.
   * Relies on the ADB controller sending and receiving data, thereby
   * generating shift register interrupts (SR_INT) for us. This means there has
   * to be activity on the ADB bus. The chip will poll to achieve this.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
347
   *
b4d76c28e   Finn Thain   macintosh/via-mac...
348
349
350
351
352
353
354
355
356
357
358
   * The VIA Port B output signalling works as follows. After the ADB transceiver
   * sees a transition on the PB4 and PB5 lines it will crank over the VIA shift
   * register which eventually raises the SR_INT interrupt. The PB4/PB5 outputs
   * are toggled with each byte as the ADB transaction progresses.
   *
   * Request with no reply expected (and empty transceiver buffer):
   *     CMD -> IDLE
   * Request with expected reply packet (or with buffered autopoll packet):
   *     CMD -> EVEN -> ODD -> EVEN -> ... -> IDLE
   * Unsolicited packet:
   *     IDLE -> EVEN -> ODD -> EVEN -> ... -> IDLE
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
359
   */
7d12e780e   David Howells   IRQ: Maintain reg...
360
  static irqreturn_t macii_interrupt(int irq, void *arg)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
361
  {
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
362
  	int x;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
363
  	struct adb_request *req;
5ce6185c2   Finn Thain   macintosh/via-mac...
364
365
366
  	unsigned long flags;
  
  	local_irq_save(flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
367

d95fd5fce   Finn Thain   m68k: Mac II ADB ...
368
369
370
371
  	if (!arg) {
  		/* Clear the SR IRQ flag when polling. */
  		if (via[IFR] & SR_INT)
  			via[IFR] = SR_INT;
5ce6185c2   Finn Thain   macintosh/via-mac...
372
373
  		else {
  			local_irq_restore(flags);
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
374
  			return IRQ_NONE;
5ce6185c2   Finn Thain   macintosh/via-mac...
375
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
376
  	}
47fd20606   Finn Thain   macintosh/via-mac...
377
  	status = via[B] & (ST_MASK | CTLR_IRQ);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
378
379
  
  	switch (macii_state) {
47fd20606   Finn Thain   macintosh/via-mac...
380
  	case idle:
b4d76c28e   Finn Thain   macintosh/via-mac...
381
382
383
  		WARN_ON((status & ST_MASK) != ST_IDLE);
  
  		reply_ptr = reply_buf;
f87a16257   Finn Thain   macintosh/via-mac...
384
  		reading_reply = false;
b4d76c28e   Finn Thain   macintosh/via-mac...
385
386
387
  
  		bus_timeout = false;
  		srq_asserted = false;
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
388

47fd20606   Finn Thain   macintosh/via-mac...
389
  		x = via[SR];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
390

b4d76c28e   Finn Thain   macintosh/via-mac...
391
392
393
  		if (!(status & CTLR_IRQ)) {
  			/* /CTLR_IRQ asserted in idle state means we must
  			 * read an autopoll reply from the transceiver buffer.
47fd20606   Finn Thain   macintosh/via-mac...
394
  			 */
47fd20606   Finn Thain   macintosh/via-mac...
395
396
397
  			macii_state = reading;
  			*reply_ptr = x;
  			reply_len = 1;
b4d76c28e   Finn Thain   macintosh/via-mac...
398
399
  		} else {
  			/* bus timeout */
b4d76c28e   Finn Thain   macintosh/via-mac...
400
  			reply_len = 0;
b16b67689   Finn Thain   macintosh/via-mac...
401
  			break;
47fd20606   Finn Thain   macintosh/via-mac...
402
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
403

47fd20606   Finn Thain   macintosh/via-mac...
404
405
406
  		/* set ADB state = even for first data byte */
  		via[B] = (via[B] & ~ST_MASK) | ST_EVEN;
  		break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
407

47fd20606   Finn Thain   macintosh/via-mac...
408
409
  	case sending:
  		req = current_req;
b4d76c28e   Finn Thain   macintosh/via-mac...
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
  
  		if (status == (ST_CMD | CTLR_IRQ)) {
  			/* /CTLR_IRQ de-asserted after the command byte means
  			 * the host can continue with the transaction.
  			 */
  
  			/* Store command byte */
  			last_cmd = req->data[1];
  			if ((last_cmd & OP_MASK) == TALK) {
  				last_talk_cmd = last_cmd;
  				if ((last_cmd & CMD_MASK) == ADB_READREG(0, 0))
  					last_poll_cmd = last_cmd;
  			}
  		}
  
  		if (status == ST_CMD) {
  			/* /CTLR_IRQ asserted after the command byte means we
  			 * must read an autopoll reply. The first byte was
  			 * lost because the shift register was an output.
  			 */
  			macii_state = reading;
f87a16257   Finn Thain   macintosh/via-mac...
431
  			reading_reply = false;
b4d76c28e   Finn Thain   macintosh/via-mac...
432
433
434
435
436
437
438
439
  			reply_ptr = reply_buf;
  			*reply_ptr = last_talk_cmd;
  			reply_len = 1;
  
  			/* reset to shift in */
  			via[ACR] &= ~SR_OUT;
  			x = via[SR];
  		} else if (data_index >= req->nbytes) {
47fd20606   Finn Thain   macintosh/via-mac...
440
  			req->sent = 1;
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
441

47fd20606   Finn Thain   macintosh/via-mac...
442
  			if (req->reply_expected) {
b4d76c28e   Finn Thain   macintosh/via-mac...
443
  				macii_state = reading;
f87a16257   Finn Thain   macintosh/via-mac...
444
  				reading_reply = true;
b4d76c28e   Finn Thain   macintosh/via-mac...
445
446
447
448
449
450
  				reply_ptr = req->reply;
  				*reply_ptr = req->data[1];
  				reply_len = 1;
  
  				via[ACR] &= ~SR_OUT;
  				x = via[SR];
624cf5b53   Finn Thain   macintosh/via-mac...
451
452
  			} else if ((req->data[1] & OP_MASK) == TALK) {
  				macii_state = reading;
f87a16257   Finn Thain   macintosh/via-mac...
453
  				reading_reply = false;
624cf5b53   Finn Thain   macintosh/via-mac...
454
455
456
457
458
459
460
461
462
463
464
  				reply_ptr = reply_buf;
  				*reply_ptr = req->data[1];
  				reply_len = 1;
  
  				via[ACR] &= ~SR_OUT;
  				x = via[SR];
  
  				req->complete = 1;
  				current_req = req->next;
  				if (req->done)
  					(*req->done)(req);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
465
  			} else {
b4d76c28e   Finn Thain   macintosh/via-mac...
466
  				macii_state = idle;
47fd20606   Finn Thain   macintosh/via-mac...
467
468
469
470
  				req->complete = 1;
  				current_req = req->next;
  				if (req->done)
  					(*req->done)(req);
b4d76c28e   Finn Thain   macintosh/via-mac...
471
  				break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
472
  			}
47fd20606   Finn Thain   macintosh/via-mac...
473
474
  		} else {
  			via[SR] = req->data[data_index++];
b4d76c28e   Finn Thain   macintosh/via-mac...
475
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
476

b4d76c28e   Finn Thain   macintosh/via-mac...
477
478
479
480
481
482
  		if ((via[B] & ST_MASK) == ST_CMD) {
  			/* just sent the command byte, set to EVEN */
  			via[B] = (via[B] & ~ST_MASK) | ST_EVEN;
  		} else {
  			/* invert state bits, toggle ODD/EVEN */
  			via[B] ^= ST_MASK;
47fd20606   Finn Thain   macintosh/via-mac...
483
484
  		}
  		break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
485

47fd20606   Finn Thain   macintosh/via-mac...
486
487
488
489
  	case reading:
  		x = via[SR];
  		WARN_ON((status & ST_MASK) == ST_CMD ||
  			(status & ST_MASK) == ST_IDLE);
47fd20606   Finn Thain   macintosh/via-mac...
490
  		if (!(status & CTLR_IRQ)) {
b4d76c28e   Finn Thain   macintosh/via-mac...
491
492
493
494
495
  			if (status == ST_EVEN && reply_len == 1) {
  				bus_timeout = true;
  			} else if (status == ST_ODD && reply_len == 2) {
  				srq_asserted = true;
  			} else {
b16b67689   Finn Thain   macintosh/via-mac...
496
497
498
499
500
501
502
503
504
505
506
507
508
509
  				macii_state = idle;
  
  				if (bus_timeout)
  					reply_len = 0;
  
  				if (reading_reply) {
  					struct adb_request *req = current_req;
  
  					req->reply_len = reply_len;
  
  					req->complete = 1;
  					current_req = req->next;
  					if (req->done)
  						(*req->done)(req);
624cf5b53   Finn Thain   macintosh/via-mac...
510
511
512
  				} else if (reply_len && autopoll_devs &&
  					   reply_buf[0] == last_poll_cmd) {
  					adb_input(reply_buf, reply_len, 1);
b16b67689   Finn Thain   macintosh/via-mac...
513
514
  				}
  				break;
47fd20606   Finn Thain   macintosh/via-mac...
515
516
  			}
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
517

b16b67689   Finn Thain   macintosh/via-mac...
518
  		if (reply_len < ARRAY_SIZE(reply_buf)) {
47fd20606   Finn Thain   macintosh/via-mac...
519
520
521
522
  			reply_ptr++;
  			*reply_ptr = x;
  			reply_len++;
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
523

47fd20606   Finn Thain   macintosh/via-mac...
524
525
526
  		/* invert state bits, toggle ODD/EVEN */
  		via[B] ^= ST_MASK;
  		break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
527

b16b67689   Finn Thain   macintosh/via-mac...
528
529
530
  	default:
  		break;
  	}
47fd20606   Finn Thain   macintosh/via-mac...
531

b16b67689   Finn Thain   macintosh/via-mac...
532
  	if (macii_state == idle) {
f93bfeb55   Finn Thain   macintosh/via-mac...
533
  		if (!current_req)
47fd20606   Finn Thain   macintosh/via-mac...
534
535
536
537
  			macii_queue_poll();
  
  		if (current_req)
  			macii_start();
47fd20606   Finn Thain   macintosh/via-mac...
538

b16b67689   Finn Thain   macintosh/via-mac...
539
540
541
  		if (macii_state == idle) {
  			via[ACR] &= ~SR_OUT;
  			x = via[SR];
47fd20606   Finn Thain   macintosh/via-mac...
542
  			via[B] = (via[B] & ~ST_MASK) | ST_IDLE;
b16b67689   Finn Thain   macintosh/via-mac...
543
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
544
  	}
d95fd5fce   Finn Thain   m68k: Mac II ADB ...
545

5ce6185c2   Finn Thain   macintosh/via-mac...
546
  	local_irq_restore(flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
547
548
  	return IRQ_HANDLED;
  }