ace_ebus.c revision 1.3 1 /* $NetBSD: ace_ebus.c,v 1.3 2011/11/19 22:51:19 tls Exp $ */
2
3 /*-
4 * Copyright (c) 2010 The NetBSD Foundation, Inc.
5 * All rights reserved.
6 *
7 * This code was written by Alessandro Forin and Neil Pittman
8 * at Microsoft Research and contributed to The NetBSD Foundation
9 * by Microsoft Corporation.
10 *
11 * Redistribution and use in source and binary forms, with or without
12 * modification, are permitted provided that the following conditions
13 * are met:
14 * 1. Redistributions of source code must retain the above copyright
15 * notice, this list of conditions and the following disclaimer.
16 * 2. Redistributions in binary form must reproduce the above copyright
17 * notice, this list of conditions and the following disclaimer in the
18 * documentation and/or other materials provided with the distribution.
19 *
20 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
21 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
22 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
23 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
24 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
25 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30 * POSSIBILITY OF SUCH DAMAGE.
31 */
32
33 #include <sys/cdefs.h>
34 __KERNEL_RCSID(0, "$NetBSD: ace_ebus.c,v 1.3 2011/11/19 22:51:19 tls Exp $");
35
36 #include "rnd.h"
37
38 #include <sys/param.h>
39 #include <sys/systm.h>
40 #include <sys/kernel.h>
41 #include <sys/conf.h>
42 #include <sys/file.h>
43 #include <sys/stat.h>
44 #include <sys/ioctl.h>
45 #include <sys/buf.h>
46 #include <sys/bufq.h>
47 #include <sys/uio.h>
48 #include <sys/malloc.h>
49 #include <sys/device.h>
50 #include <sys/disklabel.h>
51 #include <sys/disk.h>
52 #include <sys/syslog.h>
53 #include <sys/proc.h>
54 #include <sys/vnode.h>
55 #include <sys/kthread.h>
56 #include <sys/lock.h>
57 #include <sys/queue.h>
58
59 #if NRND > 0
60 #include <sys/rnd.h>
61 #endif
62
63 #include <machine/intr.h>
64 #include <machine/bus.h>
65
66 #include "locators.h"
67 #include <prop/proplib.h>
68
69 #include <emips/ebus/ebusvar.h>
70 #include <emips/emips/machdep.h>
71 #include <machine/emipsreg.h>
72
73 /* Structure returned by the Identify command (see CFlash specs)
74 * NB: We only care for the first sector so that is what we define here.
75 * NB: Beware of mis-alignment for all 32bit things
76 */
77 typedef struct _CFLASH_IDENTIFY {
78 uint16_t Signature; /* Word 0 */
79 #define CFLASH_SIGNATURE 0x848a
80 uint16_t DefaultNumberOfCylinders; /* Word 1 */
81 uint16_t Reserved1; /* Word 2 */
82 uint16_t DefaultNumberOfHeads; /* Word 3 */
83 uint16_t Obsolete1[2]; /* Word 4 */
84 uint16_t DefaultSectorsPerTrack; /* Word 6 */
85 uint16_t SectorsPerCard[2]; /* Word 7 */
86 uint16_t Obsolete2; /* Word 9 */
87 uint8_t SerialNumber[20]; /* padded, right-justified Word 10 */
88 uint16_t Obsolete3[2]; /* Word 20 */
89 uint16_t EccBytesInRWLong; /* Word 22 */
90 uint8_t FirmwareRevision[8]; /* Word 23 */
91 uint8_t ModelNumber[40]; /* Word 27 */
92 uint16_t SectorsInRWMultiple; /* Word 47 */
93 uint16_t Reserved2; /* Word 48 */
94 uint16_t Capabilities; /* Word 49 */
95 uint16_t Reserved3; /* Word 50 */
96 uint16_t PioMode; /* Word 51 */
97 uint16_t Obsolete4; /* Word 52 */
98 uint16_t FieldValidity; /* Word 53 */
99 uint16_t CurrentNumberOfCylinders; /* Word 54 */
100 uint16_t CurrentNumberOfHeads; /* Word 55 */
101 uint16_t CurrentSectorsPerTrack; /* Word 56 */
102 uint16_t CurrentCapacity[2]; /* Word 57 */
103 uint16_t MultiSectorSettings; /* Word 59 */
104 uint16_t NumberOfAddressableSectors[2]; /* Word 60 */
105 uint16_t Reserved4; /* Word 62 */
106 uint16_t MultiWordDmaTransfer; /* Word 63 */
107 uint16_t AdvancedPioModes; /* Word 64 */
108 uint16_t MinimumMultiWordDmaTiming; /* Word 65 */
109 uint16_t RecommendedMultiWordDmaTiming; /* Word 66 */
110 uint16_t PioTimingNoFlowControl; /* Word 67 */
111 uint16_t PioTimingWithFlowControl; /* Word 68 */
112 uint16_t Reserved5[13]; /* Word 69 */
113 uint16_t FeaturesSupported[3]; /* Word 82 */
114 uint16_t FeaturesEnabled[3]; /* Word 85 */
115 uint16_t UdmaMode; /* Word 88 */
116 uint16_t SecurityEraseTime; /* Word 89 */
117 uint16_t EnhancedSecurityEraseTime; /* Word 90 */
118 uint16_t CurrentPowerManagementValue; /* Word 91 */
119 uint8_t Reserved6[72]; /* Word 92-127 */
120 uint8_t SecondHalf[256]; /* Word 128-255 */
121 } CFLASH_IDENTIFY, *PCFLASH_IDENTIFY;
122
123 #define SIZEOF_IDENTIFY CF_SECTOR_SIZE /* must be a sector multiple */
124
125 /* Instead of dragging in atavar.h.. */
126 /*
127 * Parameters/state needed by the controller to perform an ATA bio.
128 */
129 struct ace_bio {
130 volatile int flags;/* cmd flags */
131 #define ATA_POLL 0x0002 /* poll for completion */
132 #define ATA_SINGLE 0x0008 /* transfer must be done in singlesector mode */
133 #define ATA_READ 0x0020 /* transfer is a read (otherwise a write) */
134 #define ATA_CORR 0x0040 /* transfer had a corrected error */
135 daddr_t blkno; /* block addr */
136 daddr_t blkdone;/* number of blks transferred */
137 size_t nblks; /* number of blocks currently transferring */
138 size_t nbytes; /* number of bytes currently transferring */
139 char *databuf;/* data buffer address */
140 volatile int error;
141 #define NOERROR 0 /* There was no error (r_error invalid),
142 else see acedone()*/
143 #define FAILED(er) (er != 0)
144 #define EDOOFUS EIO
145
146 uint32_t r_error;/* copy of status register */
147 #ifdef HAS_BAD144_HANDLING
148 daddr_t badsect[127];/* 126 plus trailing -1 marker */
149 #endif
150 };
151 /* End of atavar.h*/
152
153 struct ace_softc {
154 /* General disk infos */
155 device_t sc_dev;
156
157 struct disk sc_dk;
158 struct bufq_state *sc_q;
159 struct callout sc_restart_ch;
160
161 /* IDE disk soft states */
162 struct buf *sc_bp; /* buf being transfered */
163 struct buf *active_xfer; /* buf handoff to thread */
164 /* current transfer data */
165 struct ace_bio sc_bio; /* current transfer */
166
167 struct proc *ch_thread;
168 int ch_flags;
169 #define ATACH_SHUTDOWN 0x02 /* thread is shutting down */
170 #define ATACH_IRQ_WAIT 0x10 /* thread is waiting for irq */
171 #define ATACH_DISABLED 0x80 /* channel is disabled */
172 #define ATACH_TH_RUN 0x100 /* the kernel thread is working */
173 #define ATACH_TH_RESET 0x200 /* someone ask the thread to reset */
174
175 int openings;
176 int media_has_changed;
177 #define ACECE_MC 0x20 /* media changed */
178 #define ACECE_MCR 0x08 /* media change requested */
179 struct _CFLASH_IDENTIFY sc_params;/* drive characteristics found */
180
181 int sc_flags;
182 #define ACEF_WLABEL 0x004 /* label is writable */
183 #define ACEF_LABELLING 0x008 /* writing label */
184 #define ACEF_LOADED 0x010 /* parameters loaded */
185 #define ACEF_WAIT 0x020 /* waiting for resources */
186 #define ACEF_KLABEL 0x080 /* retain label after 'full' close */
187
188 uint64_t sc_capacity;
189 uint32_t sc_multi; /* max sectors per xfer */
190
191 struct _Sac *sc_dr; /* reg pointers */
192 int hw_busy;
193 int retries; /* number of xfer retry */
194
195 #if NRND > 0
196 krndsource_t rnd_source;
197 #endif
198 };
199
200 int ace_ebus_match(device_t, cfdata_t, void *);
201 void ace_ebus_attach(device_t, device_t, void *);
202 void aceattach(struct ace_softc *);
203 int acedetach(device_t, int);
204 int aceactivate(device_t, enum devact);
205
206 void acedone(struct ace_softc *);
207 static void ace_params_to_properties(struct ace_softc *ace);
208
209 CFATTACH_DECL_NEW(ace_ebus, sizeof(struct ace_softc),
210 ace_ebus_match, ace_ebus_attach, acedetach, aceactivate);
211
212 int ace_ebus_intr(void *cookie, void *f);
213
214 static void sysace_thread(void *arg);
215
216 int
217 ace_ebus_match(device_t parent, cfdata_t cf, void *aux)
218 {
219 struct ebus_attach_args *d = aux;
220 struct _Sac *sac = (struct _Sac *)d->ia_vaddr;
221
222 if (strcmp("ace", d->ia_name) != 0)
223 return 0;
224 if ((sac == NULL) ||
225 ((sac->Tag & SAC_TAG) != PMTTAG_SYSTEM_ACE))
226 return 0;
227 return 1;
228 }
229
230 void
231 ace_ebus_attach(device_t parent, device_t self, void *aux)
232 {
233 struct ace_softc *ace = device_private(self);
234 struct ebus_attach_args *ia = aux;
235 int error;
236
237 /*
238 * It's on the baseboard, with a dedicated interrupt line.
239 */
240 ace->sc_dr = (struct _Sac *)ia->ia_vaddr;
241 ace->sc_dev = self;
242 #if DEBUG
243 printf(" virt=%p", (void*)ace->sc_dr);
244 #endif
245 printf(" : System ACE\n");
246
247 ebus_intr_establish(parent, (void*)ia->ia_cookie, IPL_BIO,
248 ace_ebus_intr, ace);
249
250 config_pending_incr();
251
252 error = kthread_create(PRI_NONE, 0, NULL, sysace_thread,
253 ace, NULL, "%s", device_xname(ace->sc_dev));
254 if (error)
255 aprint_error_dev(ace->sc_dev, "unable to create kernel "
256 "thread: error %d\n", error);
257 }
258
259 /*
260 * Sysace driver I(af) wrote for FreeBsd.
261 */
262 #define CF_SECBITS 9
263 #define CF_SECTOR_SIZE (1 << CF_SECBITS)
264
265 static int sysace_attach(struct ace_softc *sc);
266 static int sysace_reset(struct ace_softc *sc);
267 static int sysace_identify(struct ace_softc *sc);
268 static int sysace_lock_registers(struct ace_softc *sc);
269 static int sysace_unlock_registers(struct ace_softc *sc);
270 static int sysace_start(struct ace_softc *sc, uint32_t Command, uint32_t Lba,
271 uint32_t nSectors);
272 static int sysace_validate(struct ace_softc *sc, daddr_t start, size_t *pSize);
273 static int sysace_read_at (struct ace_softc *sc, daddr_t start_sector,
274 char *buffer, size_t nblocks, size_t * pSizeRead);
275 static int sysace_write_at(struct ace_softc *sc, daddr_t start_sector,
276 char *buffer, size_t nblocks, size_t * pSizeWritten);
277 #ifdef USE_ACE_FOR_RECONFIG /* Old code, despised and replaced by ICAP */
278 static int sysace_send_config(struct ace_softc *sc,
279 uint32_t *Data, unsigned int nBytes);
280 #endif
281
282 #define DEBUG_INTR 0x01
283 #define DEBUG_XFERS 0x02
284 #define DEBUG_STATUS 0x04
285 #define DEBUG_FUNCS 0x08
286 #define DEBUG_PROBE 0x10
287 #define DEBUG_WRITES 0x20
288 #define DEBUG_READS 0x40
289 #define DEBUG_ERRORS 0x80
290 #ifdef DEBUG
291 int ace_debug = DEBUG_ERRORS /*|DEBUG_WRITES*/;
292 #define ACE_DEBUG(x) (ace_debug & (x))
293 #define DBGME(_lev_,_x_) if ((_lev_) & ace_debug) _x_
294 #else
295 #define ACE_DEBUG(x) (0)
296 #define DBGME(_lev_,_x_)
297 #endif
298 #define DEBUG_PRINT(_args_,_lev_) DBGME(_lev_,printf _args_)
299
300 static int
301 sysace_attach(struct ace_softc *sc)
302 {
303 int error;
304
305 DBGME(DEBUG_FUNCS, printf("Sysace::delayed_attach %p\n", sc));
306
307 sc->media_has_changed = TRUE;
308 sc->sc_capacity = 0;
309
310 error = sysace_reset(sc);
311 if (error) {
312 device_printf(sc->sc_dev,
313 "failed to reset, errno=%d\n", error);
314 goto Out;
315 }
316
317 error = sysace_identify(sc);
318 if (error) {
319 device_printf(sc->sc_dev,
320 "failed to identify card, errno=%d.\n", error);
321 goto Out;
322 }
323
324 DBGME(DEBUG_PROBE, device_printf(sc->sc_dev,
325 "Card has %qx sectors.\n", sc->sc_capacity));
326 if (sc->sc_capacity == 0) {
327 device_printf(sc->sc_dev, "size 0, no card? Wont work.\n");
328 error = EDOOFUS;
329 goto Out;
330 }
331
332 sc->media_has_changed = FALSE;
333 Out:
334 return error;
335 }
336
337 static void
338 sysace_wedges(void *arg);
339 extern int dkwedge_autodiscover;
340
341 /*
342 * Aux temp thread to avoid deadlock when doing
343 * the partitio.. ahem wedges thing.
344 */
345 static void
346 sysace_wedges(void *arg)
347 {
348 struct ace_softc *sc = arg;
349
350 DBGME(DEBUG_STATUS, printf("Sysace::wedges started for %p\n", sc));
351
352 /* Discover wedges on this disk. */
353 dkwedge_autodiscover = 1;
354 dkwedge_discover(&sc->sc_dk);
355
356 config_pending_decr();
357
358 DBGME(DEBUG_STATUS, printf("Sysace::thread done for %p\n", sc));
359 kthread_exit(0);
360 }
361
362 static void
363 sysace_thread(void *arg)
364 {
365 struct ace_softc *sc = arg;
366 struct buf *bp;
367 int s, error;
368
369 DBGME(DEBUG_STATUS, printf("Sysace::thread started for %p\n", sc));
370
371 s = splbio();
372 aceattach(sc);
373 splx(s);
374
375 error = kthread_create(PRI_NONE, 0 /* MPSAFE??? */, NULL,
376 sysace_wedges, sc, NULL, "%s.wedges", device_xname(sc->sc_dev));
377 if (error)
378 aprint_error_dev(sc->sc_dev, "wedges: unable to create "
379 "kernel thread: error %d\n", error);
380
381 DBGME(DEBUG_STATUS,
382 printf("Sysace::thread service active for %p\n", sc));
383
384 s = splbio();
385 for (;;) {
386 /* Get next I/O request, wait if necessary */
387 if ((sc->ch_flags & (ATACH_TH_RESET | ATACH_SHUTDOWN)) == 0 &&
388 (sc->active_xfer == NULL)) {
389 sc->ch_flags &= ~ATACH_TH_RUN;
390 (void) tsleep(&sc->ch_thread, PRIBIO, "aceth", 0);
391 sc->ch_flags |= ATACH_TH_RUN;
392 }
393 if (sc->ch_flags & ATACH_SHUTDOWN)
394 break;
395 bp = sc->active_xfer;
396 sc->active_xfer = NULL;
397 if (bp != NULL) {
398 size_t sz, bnow;
399
400 DBGME(DEBUG_XFERS,
401 printf("Sysace::task %p %p %x %p %qx %d (%zd)\n",
402 sc, bp, sc->sc_bio.flags, sc->sc_bio.databuf,
403 sc->sc_bio.blkno, sc->sc_bio.nbytes,
404 sc->sc_bio.nblks));
405
406 sc->sc_bio.error = 0;
407 for (; sc->sc_bio.nblks > 0;) {
408
409 bnow = sc->sc_bio.nblks;
410 if (sc->sc_bio.flags & ATA_SINGLE)
411 bnow = 1;
412
413 if (sc->sc_bio.flags & ATA_READ) {
414 sc->sc_bio.error =
415 sysace_read_at(sc,
416 sc->sc_bio.blkno,
417 sc->sc_bio.databuf, bnow, &sz);
418 } else {
419 sc->sc_bio.error =
420 sysace_write_at(sc,
421 sc->sc_bio.blkno,
422 sc->sc_bio.databuf, bnow, &sz);
423 }
424
425 if (FAILED(sc->sc_bio.error))
426 break;
427
428 sc->sc_bio.blkno += sz; /* in blocks */
429 sc->sc_bio.nblks -= sz;
430 sc->sc_bio.blkdone += sz;
431 sz = sz << CF_SECBITS; /* in bytes */
432 sc->sc_bio.databuf += sz;
433 sc->sc_bio.nbytes -= sz;
434 }
435
436 acedone(sc);
437 }
438 }
439
440 splx(s);
441 sc->ch_thread = NULL;
442 wakeup(&sc->ch_flags);
443 kthread_exit(0);
444 }
445
446 /* Worker routines
447 */
448 #if _DEBUG
449 typedef char *NAME;
450 typedef struct _REGDESC {
451 NAME RegisterName;
452 NAME BitNames[32];
453 } REGDESC, *PREGDESC;
454
455 static void
456 SysacePrintRegister(const REGDESC *Desc, uint32_t Value)
457 {
458 int i;
459
460 printf("\t%s %x =", Desc->RegisterName, Value);
461 for (i = 31; i >= 0; i--) {
462 if (Value & (1 << i))
463 printf(" %s",
464 (Desc->BitNames[i]) ? Desc->BitNames[i] : "?");
465 }
466 printf("\n");
467 }
468
469 static uint32_t
470 SysaceDumpRegisters(struct _Sac *regs)
471 {
472 const REGDESC Control_Names = {
473 "Control",
474 {
475 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
476 "RST", /* 0x00010000 */
477 "BUS8", /* 0x00020000 */
478 "BUS16", /* 0x00040000 */
479 "BUS32", /* 0x00080000 */
480 "IRQ", /* 0x00100000 */
481 "BRDY", /* 0x00200000 */
482 "IMSK0", /* 0x00400000 */
483 "IMSK1", /* 0x00800000 */
484 "TD0", /* 0x0f000000 */
485 "TD1", /* 0x0f000000 */
486 "TD2", /* 0x0f000000 */
487 "TD3", /* 0x0f000000 */
488 "BUFW8", /* 0x10000000 */
489 "BUFW16", /* 0x20000000 */
490 "BUFW32", /* 0x40000000 */
491 "DEBUG" /* 0x80000000 */
492 }
493 };
494
495 const REGDESC STATUS_Names = {
496 "STATUS",
497 {
498 "CFGLOCK", /* 0x00000001 */
499 "MPULOCK", /* 0x00000002 */
500 "CFGERROR", /* 0x00000004 */
501 "CFCERROR", /* 0x00000008 */
502 "CFDETECT", /* 0x00000010 */
503 "DATABUFRDY", /* 0x00000020 */
504 "DATABUFWRITE", /* 0x00000040 */
505 "CFGDONE", /* 0x00000080 */
506 "RDYFORCFCMD", /* 0x00000100 */
507 "CFGMODEPIN", /* 0x00000200 */
508 0,0,0,
509 "CFGADDRPIN0", /* 0x0000e000 */
510 "CFGADDRPIN1", /* 0x0000e000 */
511 "CFGADDRPIN2", /* 0x0000e000 */
512 0,
513 "CFBSY", /* 0x00020000 */
514 "CFRDY", /* 0x00040000 */
515 "CFDWF", /* 0x00080000 */
516 "CFDSC", /* 0x00100000 */
517 "CFDRQ", /* 0x00200000 */
518 "CFCORR", /* 0x00400000 */
519 "CFERR", /* 0x00800000 */
520 0,
521 }
522 };
523
524 const REGDESC ERRORREG_Names = {
525 "ERRORREG",
526 {
527 "CARDRESETERR", /* 0x00000001 */
528 "CARDRDYERR", /* 0x00000002 */
529 "CARDREADERR", /* 0x00000004 */
530 "CARDWRITEERR", /* 0x00000008 */
531 "SECTORRDYERR", /* 0x00000010 */
532 "CFGADDRERR", /* 0x00000020 */
533 "CFGFAILED", /* 0x00000040 */
534 "CFGREADERR", /* 0x00000080 */
535 "CFGINSTRERR", /* 0x00000100 */
536 "CFGINITERR", /* 0x00000200 */
537 0,
538 "CFBBK", /* 0x00000800 */
539 "CFUNC", /* 0x00001000 */
540 "CFIDNF", /* 0x00002000 */
541 "CFABORT", /* 0x00004000 */
542 "CFAMNF", /* 0x00008000 */
543 0,
544 }
545 };
546
547 const NAME CommandNames[8] = {
548 "0", /* 0x0000 */
549 "RESETMEMCARD", /* 0x0100 */
550 "IDENTIFYMEMCARD", /* 0x0200 */
551 "READMEMCARDDATA", /* 0x0300 */
552 "WRITEMEMCARDDATA", /* 0x0400 */
553 "5", /* 0x0500 */
554 "ABORT", /* 0x0600 */
555 "7" /* 0x0700 */
556 };
557
558 const REGDESC CONTROLREG_Names = {
559 "CONTROLREG",
560 {
561 "FORCELOCKREQ", /* 0x00000001 */
562 "LOCKREQ", /* 0x00000002 */
563 "FORCECFGADDR", /* 0x00000004 */
564 "FORCECFGMODE", /* 0x00000008 */
565 "CFGMODE", /* 0x00000010 */
566 "CFGSTART", /* 0x00000020 */
567 "CFGSEL_MPU", /* 0x00000040 */
568 "CFGRESET", /* 0x00000080 */
569 "DATABUFRDYIRQ",/* 0x00000100 */
570 "ERRORIRQ", /* 0x00000200 */
571 "CFGDONEIRQ", /* 0x00000400 */
572 "RESETIRQ", /* 0x00000800 */
573 "CFGPROG", /* 0x00001000 */
574 "CFGADDR_B0", /* 0x00002000 */
575 "CFGADDR_B1", /* 0x00004000 */
576 "CFGADDR_B2", /* 0x00008000 */
577 0,
578 }
579 };
580
581 const REGDESC FATSTATREG_Names = {
582 "FATSTATREG",
583 {
584 "MBRVALID", /* 0x00000001 */
585 "PBRVALID", /* 0x00000002 */
586 "MBRFAT12", /* 0x00000004 */
587 "PBRFAT12", /* 0x00000008 */
588 "MBRFAT16", /* 0x00000010 */
589 "PBRFAT16", /* 0x00000020 */
590 "CALCFAT12", /* 0x00000040 */
591 "CALCFAT16", /* 0x00000080 */
592 0,
593 }
594 };
595
596 printf("Sysace@%p:\n", regs);
597 printf("\tTag %x\n", regs->Tag);
598 SysacePrintRegister(&Control_Names, regs->Control);
599 printf("\tBUSMODEREG %x\n", regs->BUSMODEREG);
600 SysacePrintRegister(&STATUS_Names, regs->STATUS);
601 SysacePrintRegister(&ERRORREG_Names, regs->ERRORREG);
602 printf("\tCFGLBAREG %x\n", regs->CFGLBAREG);
603 printf("\tMPULBAREG %x\n", regs->MPULBAREG);
604 printf("\tVERSIONREG %x\n", regs->VERSIONREG);
605 printf("\tSECCNTCMDREG %x = %s cnt=%d\n", regs->SECCNTCMDREG,
606 CommandNames[(regs->SECCNTCMDREG >> 8) & 7],
607 regs->SECCNTCMDREG & SAC_SECCCNT);
608 SysacePrintRegister(&CONTROLREG_Names, regs->CONTROLREG);
609 SysacePrintRegister(&FATSTATREG_Names, regs->FATSTATREG);
610
611 return 1;
612 }
613
614 #else
615 #define SysaceDumpRegisters(_c_)
616 #endif
617
618 /*
619 * Reset the device and the interface
620 */
621 static int
622 sysace_reset(struct ace_softc *sc)
623 {
624 struct _Sac *regs = sc->sc_dr;
625
626 DBGME(DEBUG_FUNCS, printf("Sysace::Reset %p\n", sc));
627
628 /* 16bit etc etc */
629 uint32_t BusMode, Control;
630
631 /* reset our interface */
632 regs->Control = SAC_RST;
633 DELAY(200);
634
635 /* repeat on both byte lanes */
636 regs->BUSMODEREG = SAC_MODE16 | (SAC_MODE16 << 8);
637 DELAY(1);
638
639 /* check what our interface does and what the SysACE expects */
640 Control = regs->Control;
641 BusMode = regs->BUSMODEREG;
642
643 /* get them to agree */
644 if (BusMode & SAC_MODE16) {
645 regs->Control = Control | SAC_BUS16;
646 regs->Control = regs->Control & ~SAC_BUS8;
647 } else {
648 regs->Control = Control | SAC_BUS8;
649 regs->Control = regs->Control & ~SAC_BUS16;
650 }
651
652 /* check that it worked */
653 BusMode = regs->BUSMODEREG;
654 Control = regs->Control;
655
656 if (((BusMode & SAC_MODE16) == 0) && ((Control & SAC_BUS8) == 0))
657 return EDOOFUS;
658 if (((BusMode & SAC_MODE16) > 0) && ((Control & SAC_BUS16) == 0))
659 return EDOOFUS;
660
661 /* interrupts off for now */
662 regs->Control &= ~SAC_INTMASK;
663 #define SAC_INTERRUPTS (SAC_DATABUFRDYIRQ | SAC_ERRORIRQ /* | SAC_CFGDONEIRQ */)
664 Control = regs->CONTROLREG;
665 Control = (Control & ~SAC_INTERRUPTS) | SAC_RESETIRQ | SAC_FORCECFGMODE;
666 regs->CONTROLREG = Control;
667 regs->CONTROLREG = Control & ~SAC_RESETIRQ;
668
669 /* no command */
670 regs->MPULBAREG = 0;
671
672 return 0;
673 }
674
675 /*
676 * Take control of the ACE datapath
677 */
678 static int
679 sysace_lock_registers(struct ace_softc *sc)
680 {
681 uint32_t Status;
682 int i;
683
684 DBGME(DEBUG_FUNCS, printf("Sysace::Lock %p\n", sc));
685
686 /*
687 * Locked already?
688 */
689 Status = sc->sc_dr->STATUS;
690 if (Status & SAC_MPULOCK)
691 return TRUE;
692
693 /*
694 * Request lock
695 */
696 sc->sc_dr->CONTROLREG |= SAC_LOCKREQ;
697
698 /*
699 * Spin a bit until we get it
700 */
701 for (i = 0; i < 200; i++) {
702 Status = sc->sc_dr->STATUS;
703 if (Status & SAC_MPULOCK)
704 return TRUE;
705 DELAY(100);
706 DBGME(DEBUG_FUNCS,
707 printf("Sysace::Lock loops.. (st=%x)\n",Status));
708 }
709
710 /*
711 * oopsie!
712 */
713 DBGME(DEBUG_ERRORS, printf("Sysace::Lock timeout (st=%x)\n",Status));
714 SysaceDumpRegisters(sc->sc_dr);
715 return FALSE;
716 }
717
718 /*
719 * Release control of the ACE datapath
720 */
721 static int
722 sysace_unlock_registers(struct ace_softc *sc)
723 {
724 uint32_t Status;
725 int i;
726
727 DBGME(DEBUG_FUNCS, printf("Sysace::Unlock %p\n", sc));
728
729 /*
730 * Clear reset
731 */
732 sc->sc_dr->CONTROLREG &= ~SAC_CFGRESET;
733
734 /*
735 * Unlocked already?
736 */
737 Status = sc->sc_dr->STATUS;
738 if ((Status & SAC_MPULOCK) == 0)
739 return TRUE;
740
741 /*
742 * Request unlock
743 */
744 sc->sc_dr->CONTROLREG &= ~SAC_LOCKREQ;
745
746 /*
747 * Spin a bit until we get it
748 */
749 for (i = 0; i < 200; i++) {
750 Status = sc->sc_dr->STATUS;
751 if ((Status & SAC_MPULOCK) == 0)
752 return TRUE;
753 DELAY(100);
754 DBGME(DEBUG_FUNCS,
755 printf("Sysace::Unlock loops.. (st=%x)\n",Status));
756 }
757
758 /*
759 * oopsie!
760 */
761 DBGME(DEBUG_ERRORS, printf("Sysace::Unlock timeout (st=%x)\n",Status));
762 SysaceDumpRegisters(sc->sc_dr);
763 return FALSE;
764 }
765
766 /*
767 * Check if the ACE is waiting for a comamnd
768 */
769 #define sysace_ready(_s_) ((_s_)->sc_dr->STATUS & SAC_RDYFORCFCMD)
770
771 /*
772 * Check if the ACE is executing a comamnd
773 */
774 #define sysace_busy(_s_) ((_s_)->sc_dr->STATUS & SAC_CFBSY)
775
776 /*
777 * Turn on interrupts from the ACE
778 */
779 #define sysace_inton(_s_) { \
780 (_s_)->sc_dr->CONTROLREG |= SAC_INTERRUPTS; \
781 (_s_)->sc_dr->Control |= SAC_INTMASK; \
782 }
783
784 /*
785 * Turn off interrupts from the ACE
786 */
787 #define sysace_intoff(_s_) { \
788 (_s_)->sc_dr->CONTROLREG &= ~SAC_INTERRUPTS; \
789 (_s_)->sc_dr->Control &= ~SAC_INTMASK; \
790 }
791
792 /*
793 * Start a command on the ACE, such as read or identify.
794 */
795 static int
796 sysace_start(struct ace_softc *sc, uint32_t Command, uint32_t Lba,
797 uint32_t nSectors)
798 {
799
800 /*
801 * Lock it if not already
802 */
803 if (!sysace_lock_registers(sc)) {
804 /* printed already */
805 return ETIMEDOUT;
806 }
807
808 /*
809 * Is there a CF inserted
810 */
811 if (!(sc->sc_dr->STATUS & SAC_CFDETECT)) {
812 /* NB: Not a failure state */
813 DBGME(DEBUG_ERRORS,
814 printf("Sysace:: no media (st=%x)\n", sc->sc_dr->STATUS));
815 if (sc->sc_capacity) {
816 sc->media_has_changed = TRUE;
817 sc->sc_capacity = 0;
818 }
819 return ENODEV;
820 }
821
822 /*
823 * Is it ready for a command
824 */
825 if (!sysace_ready(sc)) {
826 DBGME(DEBUG_ERRORS,
827 printf("Sysace:: not ready (st=%x)\n", sc->sc_dr->STATUS));
828 SysaceDumpRegisters(sc->sc_dr);
829 return EBUSY;
830 }
831
832 /*
833 * sector number and command
834 */
835 sc->sc_dr->MPULBAREG = Lba;
836 sc->sc_dr->SECCNTCMDREG =
837 (uint16_t)(Command | (nSectors & SAC_SECCCNT));
838
839 /*
840 * re-route the chip
841 * NB: The "RESET" is actually not much of a misnomer.
842 * The chip was designed for a one-shot execution at reset time,
843 * namely loading the configuration data into the FPGA. So.
844 */
845 sc->hw_busy = TRUE;
846 sc->sc_dr->CONTROLREG |= SAC_CFGRESET;
847 return 0;
848 }
849
850 /*
851 * Identify the (size of the) CompactFlash card inserted in the slot.
852 */
853 static int
854 sysace_identify(struct ace_softc *sc)
855 {
856 PCFLASH_IDENTIFY Identify = &sc->sc_params;
857 uint32_t Status = 0;
858 int i, j, error;
859
860 DBGME(DEBUG_FUNCS, printf("Sysace::Identify %p\n", sc));
861
862 /*
863 * Turn on interrupts before we start the command
864 */
865 sysace_inton(sc); /* BUGBUG we should add polling mode (for dump too) */
866
867 /*
868 * This will invalidate the ACE's current sector data
869 */
870 sc->sc_capacity = 0;
871
872 /*
873 * Get it going
874 */
875 error = sysace_start(sc, SAC_CMD_IDENTIFYMEMCARD, 0, 1);
876
877 /*
878 * Wait until its done
879 */
880 if (!FAILED(error)) {
881
882 /* Might be called during autoconf, no interrupts */
883 if (cold) {
884 do {
885 DELAY(10);
886 Status = sc->sc_dr->STATUS;
887 } while ((Status &
888 (SAC_DATABUFRDY|SAC_CFCERROR|SAC_CFGERROR)) == 0);
889 } else {
890 while (sc->hw_busy) {
891 DBGME(DEBUG_FUNCS,
892 printf("Sysace:: cwait.. (st=%x)"
893 " sizeof=%d\n",
894 sc->sc_dr->STATUS, sizeof(*Identify)));
895 error = tsleep(&sc->media_has_changed, PRIBIO,
896 "aceidfy", 0);
897 }
898 }
899
900 /*
901 * Did it work?
902 */
903 Status = sc->sc_dr->STATUS;
904
905 if (Status & SAC_DATABUFRDY) {
906
907 /*
908 * Yes, pull out all the data.
909 * NB: Until we do so the chip will not be ready for
910 * another command
911 */
912 for (i = 0; i < sizeof(*Identify); i += 4) {
913
914 /*
915 * Verify the (32-bytes) FIFO has reloaded
916 */
917 for (j = 0; j < 10; j++) {
918 Status = sc->sc_dr->STATUS;
919 if (Status & SAC_DATABUFRDY)
920 break;
921 DELAY(10);
922 }
923 if (Status & SAC_DATABUFRDY) {
924 uint32_t Data32;
925
926 /*
927 * This pulls two 16-bit words out of
928 * the FIFO.
929 * They are ordered in LE.
930 * NB: Yes this is different from
931 * regular data accesses
932 */
933 Data32 = sc->sc_dr->DATABUFREG[0];
934 #if _BYTE_ORDER == _LITTLE_ENDIAN
935 /* all is fine */
936 #else
937 Data32 =
938 (Data32 >> 16) | (Data32 << 16);
939 #endif
940 memcpy(((char *)Identify) + i,
941 &Data32, 4);
942 } else {
943 /*
944 * Ooops, what's going on here?
945 */
946 DBGME(DEBUG_ERRORS,
947 printf("Sysace::!DATABUFRDY %x\n",
948 Status));
949 error = EIO;
950 break;
951 }
952 }
953
954 /*
955 * Make sure we did ok and pick up the relevant info
956 */
957 if (Status & SAC_DATABUFRDY) {
958 DBGME(DEBUG_XFERS,
959 device_printf(sc->sc_dev,
960 "model: %.40s/%.20s\n",
961 Identify->ModelNumber,
962 Identify->SerialNumber));
963 if (Identify->Signature == CFLASH_SIGNATURE) {
964 DBGME(DEBUG_PROBE,
965 printf("Sysace::Card is"
966 " %.40s::%.20s\n",
967 Identify->ModelNumber,
968 Identify->SerialNumber));
969
970 sc->sc_capacity =
971 (Identify->SectorsPerCard[0] << 16)
972 | Identify->SectorsPerCard[1];
973 DBGME(DEBUG_PROBE,
974 printf("Sysace::sc_capacity x%qx\n",
975 sc->sc_capacity));
976 ace_params_to_properties(sc);
977 } else {
978 DBGME(DEBUG_ERRORS,
979 printf("Sysace::Bad card signature?"
980 " %x != %x\n",
981 Identify->Signature,
982 CFLASH_SIGNATURE));
983 sc->sc_capacity = 0;
984 error = ENXIO;
985 }
986 } else {
987 error = ETIMEDOUT;
988 }
989 } else {
990 /*
991 * No, it did not work. Maybe there is no card inserted
992 */
993 DBGME(DEBUG_ERRORS,
994 printf("Sysace::Identify failed,"
995 " missing CFLASH card?\n"));
996 SysaceDumpRegisters(sc->sc_dr);
997 /* BUGBUG Fix the error code accordingly */
998 error = ETIMEDOUT;
999 }
1000 }
1001
1002 /* remember this jic */
1003 sc->sc_bio.r_error = Status;
1004
1005 /* Free the ACE for the JTAG, just in case */
1006 sysace_unlock_registers(sc);
1007
1008 /*
1009 * Done
1010 */
1011 return error;
1012 }
1013
1014 /*
1015 * Common code for read&write argument validation
1016 */
1017 static int
1018 sysace_validate(struct ace_softc *sc, daddr_t start, size_t *pSize)
1019 {
1020 daddr_t Size;
1021
1022 /*
1023 * Verify that we know the media size
1024 */
1025 if (sc->sc_capacity == 0) {
1026 int error = sysace_identify(sc);
1027 if (FAILED(error))
1028 return error;
1029 }
1030
1031 /*
1032 * Validate args
1033 */
1034 if (start >= sc->sc_capacity) {
1035 *pSize = 0;
1036 DBGME(DEBUG_ERRORS,
1037 printf("Sysace::ValidateArg(%qx) EOF\n", start));
1038 return E2BIG;
1039 }
1040
1041 /*
1042 * Adjust size if necessary
1043 */
1044 Size = start + *pSize;
1045 if (Size > sc->sc_capacity) {
1046 /*
1047 * At most this many sectors
1048 */
1049 Size = sc->sc_capacity - start;
1050 *pSize = (size_t)Size;
1051 }
1052
1053 DBGME(DEBUG_FUNCS,
1054 printf("Sysace::Validate %qx %zd\n", start, *pSize));
1055 return 0;
1056 }
1057
1058 /* Read SIZE bytes from sysace device, at offset Position
1059 */
1060 uint32_t ace_maxatatime = 255;
1061 #define MAXATATIME ace_maxatatime //255 /* BUGBUG test me on real hardware!! */
1062
1063 static int
1064 sysace_read_at(struct ace_softc *sc, daddr_t start_sector, char *buffer,
1065 size_t nblocks, size_t *pSizeRead)
1066 {
1067 int error;
1068 uint32_t BlocksThisTime;
1069 uint32_t Status = 0, SizeRead = 0;
1070 uint32_t i, j;
1071
1072 DBGME(DEBUG_XFERS|DEBUG_READS,
1073 printf("SysaceReadAt(%p %qx %p %zd %p)\n",
1074 sc, start_sector, buffer, nblocks, pSizeRead));
1075
1076 /*
1077 * Validate & trim arguments
1078 */
1079 error = sysace_validate(sc, start_sector, &nblocks);
1080
1081 /*
1082 * Repeat until we are done or error
1083 */
1084 while (error == 0) {
1085
1086 /*
1087 * .. one bunch of sectors at a time
1088 */
1089 BlocksThisTime = nblocks;
1090 if (BlocksThisTime > MAXATATIME)
1091 BlocksThisTime = MAXATATIME;
1092
1093 /*
1094 * Yes, start a sector read
1095 */
1096 sysace_inton(sc);
1097 error = sysace_start(sc,
1098 SAC_CMD_READMEMCARDDATA,
1099 (uint32_t)start_sector, /* BUGBUG trims here, no warn. */
1100 BlocksThisTime);
1101 /*
1102 * And wait until done, if ok
1103 */
1104 if (!FAILED(error)) {
1105 start_sector += BlocksThisTime;
1106 /* Might be called during autoconf, no interrupts */
1107 /* BUGBUG timeouts! */
1108 if (cold) {
1109 do {
1110 DELAY(10);
1111 Status = sc->sc_dr->STATUS;
1112 } while ((Status &
1113 (SAC_DATABUFRDY|SAC_CFCERROR|SAC_CFGERROR))
1114 == 0);
1115 } else {
1116 while (sc->hw_busy) {
1117 error = tsleep(&sc->media_has_changed,
1118 PRIBIO, "aceread", 0);
1119 }
1120 }
1121 }
1122
1123 /*
1124 * Are we doing ok
1125 */
1126 if (!FAILED(error)) {
1127
1128 /*
1129 * Get the data out of the ACE
1130 */
1131 for (i = 0; i < (BlocksThisTime << CF_SECBITS);
1132 i += 4) {
1133
1134 /*
1135 * Make sure the FIFO is ready
1136 */
1137 for (j = 0; j < 10; j++) {
1138 Status = sc->sc_dr->STATUS;
1139 if (Status & SAC_DATABUFRDY)
1140 break;
1141 DELAY(1000);
1142 }
1143
1144 /*
1145 * Got it?
1146 */
1147 if (Status & SAC_DATABUFRDY) {
1148 uint32_t Data32;
1149
1150 Data32 = sc->sc_dr->DATABUFREG[0];
1151 Data32 = le32toh(Data32);
1152 memcpy(buffer + i, &Data32, 4);
1153 } else {
1154 /*
1155 * Ooops, get out of here
1156 */
1157 DBGME(DEBUG_ERRORS,
1158 printf("Sysace::READ timeout\n"));
1159 SysaceDumpRegisters(sc->sc_dr);
1160 error = ETIMEDOUT;
1161 break;
1162 }
1163 }
1164
1165 /*
1166 * Still doing ok?
1167 */
1168 if (!FAILED(error)) {
1169 nblocks -= BlocksThisTime;
1170 SizeRead += BlocksThisTime;
1171 buffer += BlocksThisTime << CF_SECBITS;
1172 } else {
1173 /* remember this jic */
1174 sc->sc_bio.r_error = Status;
1175 }
1176 }
1177
1178 /* Free the ACE for the JTAG, just in case */
1179 sysace_unlock_registers(sc);
1180
1181 /*
1182 * Are we done yet?
1183 */
1184 if (nblocks == 0)
1185 break;
1186 }
1187
1188 if (pSizeRead)
1189 *pSizeRead = SizeRead;
1190 return error;
1191 }
1192
1193 /*
1194 * Write SIZE bytes to device.
1195 */
1196 static int
1197 sysace_write_at(struct ace_softc *sc, daddr_t start_sector, char *buffer,
1198 size_t nblocks, size_t *pSizeWritten)
1199 {
1200 int error;
1201 uint32_t BlocksThisTime;
1202 uint32_t Status = 0, SizeWritten = 0;
1203 uint32_t i, j;
1204
1205 DBGME(DEBUG_XFERS|DEBUG_WRITES,
1206 printf("SysaceWriteAt(%p %qx %p %zd %p)\n",
1207 sc, start_sector, buffer, nblocks, pSizeWritten));
1208
1209 /*
1210 * Validate & trim arguments
1211 */
1212 error = sysace_validate(sc, start_sector, &nblocks);
1213
1214 /*
1215 * Repeat until we are done or error
1216 */
1217 while (error == 0) {
1218
1219 /*
1220 * .. one sector at a time
1221 * BUGBUG Supposedly we can do up to 256 sectors?
1222 */
1223 BlocksThisTime = nblocks;
1224 if (BlocksThisTime > MAXATATIME)
1225 BlocksThisTime = MAXATATIME;
1226
1227 /*
1228 * Yes, start a sector write
1229 */
1230 sysace_inton(sc);
1231 error = sysace_start(sc,
1232 SAC_CMD_WRITEMEMCARDDATA,
1233 (uint32_t)start_sector, /* BUGBUG trims here, no warn. */
1234 BlocksThisTime);
1235 /*
1236 * And wait until done, if ok
1237 */
1238 if (!FAILED(error)) {
1239 start_sector += BlocksThisTime;
1240 /* BUGBUG timeouts! */
1241 while (sc->hw_busy) {
1242 error = tsleep(&sc->media_has_changed,
1243 PRIBIO, "acewrite", 0);
1244 }
1245 }
1246
1247 /*
1248 * Are we doing ok
1249 */
1250 if (!FAILED(error)) {
1251
1252 /*
1253 * Get the data out to the ACE
1254 */
1255 for (i = 0; i < (BlocksThisTime << CF_SECBITS);
1256 i += 4) {
1257
1258 /*
1259 * Make sure the FIFO is ready
1260 */
1261 for (j = 0; j < 10; j++) {
1262 Status = sc->sc_dr->STATUS;
1263 if (Status & SAC_DATABUFRDY)
1264 break;
1265 DELAY(1000);
1266 }
1267
1268 /*
1269 * Got it?
1270 */
1271 if (Status & SAC_DATABUFRDY) {
1272 uint32_t Data32;
1273
1274 memcpy(&Data32, buffer + i, 4);
1275 Data32 = htole32(Data32);
1276 sc->sc_dr->DATABUFREG[0] = Data32;
1277 } else {
1278 /*
1279 * Ooops, get out of here
1280 */
1281 DBGME(DEBUG_ERRORS,
1282 printf("Sysace::WRITE timeout\n"));
1283 SysaceDumpRegisters(sc->sc_dr);
1284 error = ETIMEDOUT;
1285 /* remember this jic */
1286 sc->sc_bio.r_error = Status;
1287 break;
1288 }
1289 }
1290
1291 /*
1292 * Still doing ok?
1293 */
1294 if (!FAILED(error)) {
1295 nblocks -= BlocksThisTime;
1296 SizeWritten += BlocksThisTime;
1297 buffer += BlocksThisTime << CF_SECBITS;
1298 }
1299 }
1300
1301 /*
1302 * We need to wait until the device is ready for the
1303 * next command
1304 * Experimentation shows that it can take longer than 10msec.
1305 */
1306 if (!FAILED(error)) {
1307 for (j = 0; j < 300; j++) {
1308 Status = sc->sc_dr->STATUS;
1309 if (Status & SAC_RDYFORCFCMD)
1310 break;
1311 (void)tsleep(&sc->media_has_changed,
1312 PRIBIO, "acewrite", 2);
1313 }
1314 if (!(Status & SAC_RDYFORCFCMD)) {
1315 DBGME(DEBUG_ERRORS,
1316 printf("Sysace::WRITE-COMPLETE timeout"
1317 " St=%x\n", Status));
1318 SysaceDumpRegisters(sc->sc_dr);
1319 /*
1320 * Ignore, we'll handle it the next time around
1321 * BUGBUG To be revised along with non-existant
1322 * error handling
1323 */
1324 }
1325 }
1326
1327 /* Free the ACE for the JTAG, just in case */
1328 sysace_unlock_registers(sc);
1329
1330 /*
1331 * Are we done yet?
1332 */
1333 if (nblocks == 0)
1334 break;
1335 }
1336
1337 if (pSizeWritten)
1338 *pSizeWritten = SizeWritten;
1339 return error;
1340 }
1341
1342 int
1343 ace_ebus_intr(void *cookie, void *f)
1344 {
1345 struct ace_softc *sc = cookie;
1346 uint32_t Control;
1347
1348 /*
1349 * Turn off interrupts and ACK them
1350 */
1351 sysace_intoff(sc);
1352
1353 Control = sc->sc_dr->CONTROLREG & (~(SAC_RESETIRQ|SAC_INTERRUPTS));
1354 sc->sc_dr->CONTROLREG = Control | SAC_RESETIRQ;
1355 sc->sc_dr->CONTROLREG = Control;
1356
1357 /* ... read status and do whatever ... */
1358
1359 sc->hw_busy = FALSE;
1360 wakeup(&sc->media_has_changed);
1361 return 1;
1362 }
1363
1364 #ifdef USE_ACE_FOR_RECONFIG
1365 static int
1366 sysace_send_config(struct ace_softc *sc, uint32_t *Data, unsigned int nBytes)
1367 {
1368 struct _Sac *Interface = sc->sc_dr;
1369 unsigned int i, j, nWords;
1370 uint32_t CtlWas;
1371 uint32_t Status;
1372
1373 CtlWas = Interface->CONTROLREG;
1374
1375 /* Set the bits but in RESET (pag 49-50 of specs)*/
1376 #define CFGCMD (SAC_FORCELOCKREQ | SAC_LOCKREQ | SAC_CFGSEL | \
1377 SAC_FORCECFGMODE |/* SAC_CFGMODE |*/ SAC_CFGSTART)
1378
1379 Interface->CONTROLREG = CFGCMD | SAC_CFGRESET;
1380
1381 /* Take it out of RESET */
1382 Interface->CONTROLREG = CFGCMD;
1383
1384 /*
1385 * Must wait till it says READY
1386 * It can take a looong time
1387 */
1388 for (j = 0; j < 1000; j++) {
1389 Status = Interface->STATUS;
1390 if (Status & SAC_RDYFORCFCMD)
1391 break;
1392 DELAY(1000);
1393 }
1394
1395 if (0 == (Status & SAC_RDYFORCFCMD)) {
1396 DBGME(DEBUG_ERRORS,
1397 printf("Sysace::CMD error %x (j=%d)\n", Status, j));
1398 goto Error;
1399 }
1400
1401 /*
1402 * Get the data out to the ACE
1403 */
1404 #define ACEROUNDUP 32
1405 nBytes = (nBytes + ACEROUNDUP - 1) & ~(ACEROUNDUP-1);
1406 nWords = (nBytes + 3) / 4;
1407 DBGME(DEBUG_FUNCS,
1408 printf("Sending %d bytes (as %d words) to %p ",
1409 nBytes, nWords, Interface));
1410 for (i = 0; i < nWords; i += 1/*word*/) {
1411
1412 /* Stop on errors */
1413 Status = Interface->ERRORREG;
1414 if (Status) {
1415 /*
1416 * Ooops, get out of here
1417 */
1418 DBGME(DEBUG_ERRORS,
1419 printf("Sysace::CFG error %x (i=%d)\n", Status, i));
1420 goto Error;
1421 }
1422
1423 /*
1424 * Make sure the FIFO is ready
1425 */
1426 for (j = 0; j < 100; j++) {
1427 Status = Interface->STATUS;
1428 if (Status & SAC_DATABUFRDY)
1429 break;
1430 DELAY(1000);
1431 }
1432
1433 /*
1434 * Got it?
1435 */
1436 if (Status & SAC_DATABUFRDY) {
1437 uint32_t Data32;
1438
1439 Data32 = Data[i];
1440 Data32 = htole32(Data32);
1441 Interface->DATABUFREG[0] = Data32;
1442 } else {
1443 /*
1444 * Ooops, get out of here
1445 */
1446 DBGME(DEBUG_ERRORS,
1447 printf("Sysace::WRITE timeout %x (i=%d)\n",
1448 Status, i));
1449 goto Error;
1450 }
1451 }
1452 DBGME(DEBUG_FUNCS, printf("done ok.\n"));
1453
1454 /* Put it back the way it was (try to.. :-( )*/
1455 Interface->CONTROLREG = CtlWas;
1456 return 0;
1457
1458 Error:
1459 SysaceDumpRegisters(Interface);
1460 Interface->CONTROLREG = CtlWas;
1461 return EIO;
1462 }
1463 #endif /* USE_ACE_FOR_RECONFIG */
1464
1465
1466 /*
1467 * Rest of code lifted with mods from the dev\ata\wd.c driver
1468 */
1469
1470 /* $NetBSD: ace_ebus.c,v 1.3 2011/11/19 22:51:19 tls Exp $ */
1471
1472 /*
1473 * Copyright (c) 1998, 2001 Manuel Bouyer. All rights reserved.
1474 *
1475 * Redistribution and use in source and binary forms, with or without
1476 * modification, are permitted provided that the following conditions
1477 * are met:
1478 * 1. Redistributions of source code must retain the above copyright
1479 * notice, this list of conditions and the following disclaimer.
1480 * 2. Redistributions in binary form must reproduce the above copyright
1481 * notice, this list of conditions and the following disclaimer in the
1482 * documentation and/or other materials provided with the distribution.
1483 * 3. All advertising materials mentioning features or use of this software
1484 * must display the following acknowledgement:
1485 * This product includes software developed by Manuel Bouyer.
1486 * 4. The name of the author may not be used to endorse or promote products
1487 * derived from this software without specific prior written permission.
1488 *
1489 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
1490 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
1491 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
1492 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
1493 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
1494 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
1495 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
1496 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
1497 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
1498 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
1499 */
1500
1501 /*-
1502 * Copyright (c) 1998, 2003, 2004 The NetBSD Foundation, Inc.
1503 * All rights reserved.
1504 *
1505 * This code is derived from software contributed to The NetBSD Foundation
1506 * by Charles M. Hannum and by Onno van der Linden.
1507 *
1508 * Redistribution and use in source and binary forms, with or without
1509 * modification, are permitted provided that the following conditions
1510 * are met:
1511 * 1. Redistributions of source code must retain the above copyright
1512 * notice, this list of conditions and the following disclaimer.
1513 * 2. Redistributions in binary form must reproduce the above copyright
1514 * notice, this list of conditions and the following disclaimer in the
1515 * documentation and/or other materials provided with the distribution.
1516 * 3. All advertising materials mentioning features or use of this software
1517 * must display the following acknowledgement:
1518 * This product includes software developed by the NetBSD
1519 * Foundation, Inc. and its contributors.
1520 * 4. Neither the name of The NetBSD Foundation nor the names of its
1521 * contributors may be used to endorse or promote products derived
1522 * from this software without specific prior written permission.
1523 *
1524 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
1525 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
1526 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
1527 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
1528 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
1529 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
1530 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
1531 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
1532 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
1533 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
1534 * POSSIBILITY OF SUCH DAMAGE.
1535 */
1536
1537 static const char ST506[] = "ST506";
1538
1539 #define ACEIORETRIES_SINGLE 4 /* number of retries before single-sector */
1540 #define ACEIORETRIES 5 /* number of retries before giving up */
1541 #define RECOVERYTIME hz/2 /* time to wait before retrying a cmd */
1542
1543 #define ACEUNIT(dev) DISKUNIT(dev)
1544 #define ACEPART(dev) DISKPART(dev)
1545 #define ACEMINOR(unit, part) DISKMINOR(unit, part)
1546 #define MAKEACEDEV(maj, unit, part) MAKEDISKDEV(maj, unit, part)
1547
1548 #define ACELABELDEV(dev) (MAKEACEDEV(major(dev), ACEUNIT(dev), RAW_PART))
1549
1550 void aceperror(const struct ace_softc *);
1551
1552 extern struct cfdriver ace_cd;
1553
1554 dev_type_open(aceopen);
1555 dev_type_close(aceclose);
1556 dev_type_read(aceread);
1557 dev_type_write(acewrite);
1558 dev_type_ioctl(aceioctl);
1559 dev_type_strategy(acestrategy);
1560 dev_type_dump(acedump);
1561 dev_type_size(acesize);
1562
1563 const struct bdevsw ace_bdevsw = {
1564 aceopen, aceclose, acestrategy, aceioctl, acedump, acesize, D_DISK
1565 };
1566
1567 const struct cdevsw ace_cdevsw = {
1568 aceopen, aceclose, aceread, acewrite, aceioctl,
1569 nostop, notty, nopoll, nommap, nokqfilter, D_DISK
1570 };
1571
1572 void acegetdefaultlabel(struct ace_softc *, struct disklabel *);
1573 void acegetdisklabel(struct ace_softc *);
1574 void acestart(void *);
1575 void __acestart(struct ace_softc*, struct buf *);
1576 void acerestart(void *);
1577
1578 struct dkdriver acedkdriver = { acestrategy, minphys };
1579
1580 #ifdef HAS_BAD144_HANDLING
1581 static void bad144intern(struct ace_softc *);
1582 #endif
1583
1584 void
1585 aceattach(struct ace_softc *ace)
1586 {
1587 struct device *self = ace->sc_dev;
1588 char tbuf[41], pbuf[9], c, *p, *q;
1589 int i, blank;
1590 DEBUG_PRINT(("aceattach\n"), DEBUG_FUNCS | DEBUG_PROBE);
1591
1592 callout_init(&ace->sc_restart_ch, 0);
1593 bufq_alloc(&ace->sc_q, BUFQ_DISK_DEFAULT_STRAT, BUFQ_SORT_RAWBLOCK);
1594
1595 ace->openings = 1; /* wazziz?*/
1596 ace->sc_multi = MAXATATIME;
1597
1598 aprint_naive("\n");
1599
1600 /* setup all required fields so that if the attach fails we are ok */
1601 ace->sc_dk.dk_driver = &acedkdriver;
1602 ace->sc_dk.dk_name = device_xname(ace->sc_dev);
1603
1604 /* read our drive info */
1605 if (sysace_attach(ace) != 0) {
1606 aprint_error_dev(ace->sc_dev, "attach failed\n");
1607 return;
1608 }
1609
1610 aprint_normal_dev(ace->sc_dev, "drive supports %d-sector PIO xfers\n",
1611 ace->sc_multi);
1612
1613 for (blank = 0, p = ace->sc_params.ModelNumber, q = tbuf, i = 0;
1614 i < sizeof(ace->sc_params.ModelNumber); i++) {
1615 c = *p++;
1616 if (c == '\0')
1617 break;
1618 if (c != ' ') {
1619 if (blank) {
1620 *q++ = ' ';
1621 blank = 0;
1622 }
1623 *q++ = c;
1624 } else
1625 blank = 1;
1626 }
1627 *q++ = '\0';
1628
1629 aprint_normal_dev(ace->sc_dev, "card is <%s>\n", tbuf);
1630
1631 format_bytes(pbuf, sizeof(pbuf), ace->sc_capacity * DEV_BSIZE);
1632 aprint_normal("%s: %s, %d cyl, %d head, %d sec, "
1633 "%d bytes/sect x %llu sectors\n",
1634 self->dv_xname, pbuf,
1635 (int)(ace->sc_capacity /
1636 (ace->sc_params.CurrentNumberOfHeads *
1637 ace->sc_params.CurrentSectorsPerTrack)),
1638 ace->sc_params.CurrentNumberOfHeads,
1639 ace->sc_params.CurrentSectorsPerTrack,
1640 DEV_BSIZE, (unsigned long long)ace->sc_capacity);
1641
1642 /*
1643 * Attach the disk structure. We fill in dk_info later.
1644 */
1645 disk_attach(&ace->sc_dk);
1646
1647 #if NRND > 0
1648 rnd_attach_source(&ace->rnd_source, device_xname(ace->sc_dev),
1649 RND_TYPE_DISK, 0);
1650 #endif
1651
1652 }
1653
1654 int
1655 aceactivate(struct device *self, enum devact act)
1656 {
1657 int rv = 0;
1658
1659 switch (act) {
1660 case DVACT_DEACTIVATE:
1661 /*
1662 * Nothing to do; we key off the device's DVF_ACTIVATE.
1663 */
1664 break;
1665 default:
1666 rv = EOPNOTSUPP;
1667 }
1668 return rv;
1669 }
1670
1671 int
1672 acedetach(struct device *self, int flags)
1673 {
1674 struct ace_softc *sc = device_private(self);
1675 int s, bmaj, cmaj, i, mn;
1676
1677 /* locate the major number */
1678 bmaj = bdevsw_lookup_major(&ace_bdevsw);
1679 cmaj = cdevsw_lookup_major(&ace_cdevsw);
1680
1681 /* Nuke the vnodes for any open instances. */
1682 for (i = 0; i < MAXPARTITIONS; i++) {
1683 mn = ACEMINOR(device_unit(self), i);
1684 vdevgone(bmaj, mn, mn, VBLK);
1685 vdevgone(cmaj, mn, mn, VCHR);
1686 }
1687
1688 /* Delete all of our wedges. */
1689 dkwedge_delall(&sc->sc_dk);
1690
1691 s = splbio();
1692
1693 /* Kill off any queued buffers. */
1694 bufq_drain(sc->sc_q);
1695
1696 bufq_free(sc->sc_q);
1697 #if 0
1698 sc->atabus->ata_killpending(sc->drvp);
1699 #endif
1700
1701 splx(s);
1702
1703 /* Detach disk. */
1704 disk_detach(&sc->sc_dk);
1705
1706 #if NRND > 0
1707 /* Unhook the entropy source. */
1708 rnd_detach_source(&sc->rnd_source);
1709 #endif
1710
1711 #if 0
1712 sc->drvp->drive_flags = 0; /* no drive any more here */
1713 #endif
1714
1715 return 0;
1716 }
1717
1718 /*
1719 * Read/write routine for a buffer. Validates the arguments and schedules the
1720 * transfer. Does not wait for the transfer to complete.
1721 */
1722 void
1723 acestrategy(struct buf *bp)
1724 {
1725 struct ace_softc *ace;
1726 struct disklabel *lp;
1727 daddr_t blkno;
1728 int s;
1729
1730 ace = device_lookup_private(&ace_cd, ACEUNIT(bp->b_dev));
1731
1732 if (ace == NULL) {
1733 bp->b_error = ENXIO;
1734 biodone(bp);
1735 return;
1736 }
1737 lp = ace->sc_dk.dk_label;
1738
1739 DEBUG_PRINT(("acestrategy (%s) %lld\n",
1740 device_xname(ace->sc_dev), bp->b_blkno), DEBUG_XFERS);
1741
1742 /* Valid request? */
1743 if (bp->b_blkno < 0 ||
1744 (bp->b_bcount % lp->d_secsize) != 0 ||
1745 (bp->b_bcount / lp->d_secsize) >= (1 << NBBY)) {
1746 bp->b_error = EINVAL;
1747 goto done;
1748 }
1749
1750 /* If device invalidated (e.g. media change, door open), error. */
1751 if ((ace->sc_flags & ACEF_LOADED) == 0) {
1752 bp->b_error = EIO;
1753 goto done;
1754 }
1755
1756 /* If it's a null transfer, return immediately. */
1757 if (bp->b_bcount == 0)
1758 goto done;
1759
1760 /*
1761 * Do bounds checking, adjust transfer. if error, process.
1762 * If end of partition, just return.
1763 */
1764 if (ACEPART(bp->b_dev) == RAW_PART) {
1765 if (bounds_check_with_mediasize(bp, DEV_BSIZE,
1766 ace->sc_capacity) <= 0)
1767 goto done;
1768 } else {
1769 if (bounds_check_with_label(&ace->sc_dk, bp,
1770 (ace->sc_flags & (ACEF_WLABEL|ACEF_LABELLING)) != 0) <= 0)
1771 goto done;
1772 }
1773
1774 /*
1775 * Now convert the block number to absolute and put it in
1776 * terms of the device's logical block size.
1777 */
1778 if (lp->d_secsize >= DEV_BSIZE)
1779 blkno = bp->b_blkno / (lp->d_secsize / DEV_BSIZE);
1780 else
1781 blkno = bp->b_blkno * (DEV_BSIZE / lp->d_secsize);
1782
1783 if (ACEPART(bp->b_dev) != RAW_PART)
1784 blkno += lp->d_partitions[ACEPART(bp->b_dev)].p_offset;
1785
1786 bp->b_rawblkno = blkno;
1787
1788 /* Queue transfer on drive, activate drive and controller if idle. */
1789 s = splbio();
1790 bufq_put(ace->sc_q, bp);
1791 acestart(ace);
1792 splx(s);
1793 return;
1794 done:
1795 /* Toss transfer; we're done early. */
1796 bp->b_resid = bp->b_bcount;
1797 biodone(bp);
1798 }
1799
1800 /*
1801 * Queue a drive for I/O.
1802 */
1803 void
1804 acestart(void *arg)
1805 {
1806 struct ace_softc *ace = arg;
1807 struct buf *bp = NULL;
1808
1809 DEBUG_PRINT(("acestart %s\n", device_xname(ace->sc_dev)), DEBUG_XFERS);
1810 while (ace->openings > 0) {
1811
1812 /* Is there a buf for us ? */
1813 if ((bp = bufq_get(ace->sc_q)) == NULL)
1814 return;
1815
1816 /*
1817 * Make the command. First lock the device
1818 */
1819 ace->openings--;
1820
1821 ace->retries = 0;
1822 __acestart(ace, bp);
1823 }
1824 }
1825
1826 void
1827 __acestart(struct ace_softc *sc, struct buf *bp)
1828 {
1829
1830 sc->sc_bp = bp;
1831 /*
1832 * If we're retrying, retry in single-sector mode. This will give us
1833 * the sector number of the problem, and will eventually allow the
1834 * transfer to succeed.
1835 */
1836 if (sc->retries >= ACEIORETRIES_SINGLE)
1837 sc->sc_bio.flags = ATA_SINGLE;
1838 else
1839 sc->sc_bio.flags = 0;
1840 if (bp->b_flags & B_READ)
1841 sc->sc_bio.flags |= ATA_READ;
1842 sc->sc_bio.blkno = bp->b_rawblkno;
1843 sc->sc_bio.blkdone = 0;
1844 sc->sc_bio.nbytes = bp->b_bcount;
1845 sc->sc_bio.nblks = bp->b_bcount >> CF_SECBITS;
1846 sc->sc_bio.databuf = bp->b_data;
1847 /* Instrumentation. */
1848 disk_busy(&sc->sc_dk);
1849 sc->active_xfer = bp;
1850 wakeup(&sc->ch_thread);
1851 }
1852
1853 void
1854 acedone(struct ace_softc *ace)
1855 {
1856 struct buf *bp = ace->sc_bp;
1857 const char *errmsg;
1858 int do_perror = 0;
1859
1860 DEBUG_PRINT(("acedone %s\n", device_xname(ace->sc_dev)), DEBUG_XFERS);
1861
1862 if (bp == NULL)
1863 return;
1864
1865 bp->b_resid = ace->sc_bio.nbytes;
1866 switch (ace->sc_bio.error) {
1867 case ETIMEDOUT:
1868 errmsg = "device timeout";
1869 do_perror = 1;
1870 goto retry;
1871 case EBUSY:
1872 case EDOOFUS:
1873 errmsg = "device stuck";
1874 retry: /* Just reset and retry. Can we do more ? */
1875 sysace_reset(ace);
1876 diskerr(bp, "ace", errmsg, LOG_PRINTF,
1877 ace->sc_bio.blkdone, ace->sc_dk.dk_label);
1878 if (ace->retries < ACEIORETRIES)
1879 printf(", retrying");
1880 printf("\n");
1881 if (do_perror)
1882 aceperror(ace);
1883 if (ace->retries < ACEIORETRIES) {
1884 ace->retries++;
1885 callout_reset(&ace->sc_restart_ch, RECOVERYTIME,
1886 acerestart, ace);
1887 return;
1888 }
1889
1890 bp->b_error = EIO;
1891 break;
1892 case 0:
1893 if ((ace->sc_bio.flags & ATA_CORR) || ace->retries > 0)
1894 printf("%s: soft error (corrected)\n",
1895 device_xname(ace->sc_dev));
1896 break;
1897 case ENODEV:
1898 case E2BIG:
1899 bp->b_error = EIO;
1900 break;
1901 }
1902 disk_unbusy(&ace->sc_dk, (bp->b_bcount - bp->b_resid),
1903 (bp->b_flags & B_READ));
1904 #if NRND > 0
1905 rnd_add_uint32(&ace->rnd_source, bp->b_blkno);
1906 #endif
1907 biodone(bp);
1908 ace->openings++;
1909 acestart(ace);
1910 }
1911
1912 void
1913 acerestart(void *v)
1914 {
1915 struct ace_softc *ace = v;
1916 struct buf *bp = ace->sc_bp;
1917 int s;
1918 DEBUG_PRINT(("acerestart %s\n",
1919 device_xname(ace->sc_dev)), DEBUG_XFERS);
1920
1921 s = splbio();
1922 __acestart(v, bp);
1923 splx(s);
1924 }
1925
1926 int
1927 aceread(dev_t dev, struct uio *uio, int flags)
1928 {
1929 int r;
1930
1931 DEBUG_PRINT(("aceread\n"), DEBUG_XFERS);
1932 r = physio(acestrategy, NULL, dev, B_READ, minphys, uio);
1933 DEBUG_PRINT(("aceread -> x%x resid=%x\n",r,uio->uio_resid),DEBUG_XFERS);
1934
1935 return r;
1936 }
1937
1938 int
1939 acewrite(dev_t dev, struct uio *uio, int flags)
1940 {
1941
1942 DEBUG_PRINT(("acewrite\n"), DEBUG_XFERS);
1943 return physio(acestrategy, NULL, dev, B_WRITE, minphys, uio);
1944 }
1945
1946 int
1947 aceopen(dev_t dev, int flag, int fmt, struct lwp *l)
1948 {
1949 struct ace_softc *ace;
1950 int part, error;
1951
1952 DEBUG_PRINT(("aceopen\n"), DEBUG_FUNCS);
1953 ace = device_lookup_private(&ace_cd, ACEUNIT(dev));
1954 if (ace == NULL)
1955 return ENXIO;
1956
1957 if (! device_is_active(ace->sc_dev))
1958 return ENODEV;
1959
1960 part = ACEPART(dev);
1961
1962 mutex_enter(&ace->sc_dk.dk_openlock);
1963
1964 /*
1965 * If there are wedges, and this is not RAW_PART, then we
1966 * need to fail.
1967 */
1968 if (ace->sc_dk.dk_nwedges != 0 && part != RAW_PART) {
1969 error = EBUSY;
1970 goto bad;
1971 }
1972
1973 if (ace->sc_dk.dk_openmask != 0) {
1974 /*
1975 * If any partition is open, but the disk has been invalidated,
1976 * disallow further opens.
1977 */
1978 if ((ace->sc_flags & ACEF_LOADED) == 0) {
1979 error = EIO;
1980 goto bad;
1981 }
1982 } else {
1983 if ((ace->sc_flags & ACEF_LOADED) == 0) {
1984 ace->sc_flags |= ACEF_LOADED;
1985
1986 /* Load the physical device parameters. */
1987 if (ace->sc_capacity == 0) {
1988 error = sysace_identify(ace);
1989 if (error)
1990 goto bad;
1991 }
1992
1993 /* Load the partition info if not already loaded. */
1994 acegetdisklabel(ace);
1995 }
1996 }
1997
1998 /* Check that the partition exists. */
1999 if (part != RAW_PART &&
2000 (part >= ace->sc_dk.dk_label->d_npartitions ||
2001 ace->sc_dk.dk_label->d_partitions[part].p_fstype == FS_UNUSED)) {
2002 error = ENXIO;
2003 goto bad;
2004 }
2005
2006 /* Insure only one open at a time. */
2007 switch (fmt) {
2008 case S_IFCHR:
2009 ace->sc_dk.dk_copenmask |= (1 << part);
2010 break;
2011 case S_IFBLK:
2012 ace->sc_dk.dk_bopenmask |= (1 << part);
2013 break;
2014 }
2015 ace->sc_dk.dk_openmask =
2016 ace->sc_dk.dk_copenmask | ace->sc_dk.dk_bopenmask;
2017
2018 mutex_exit(&ace->sc_dk.dk_openlock);
2019 return 0;
2020
2021 bad:
2022 mutex_exit(&ace->sc_dk.dk_openlock);
2023 return error;
2024 }
2025
2026 int
2027 aceclose(dev_t dev, int flag, int fmt, struct lwp *l)
2028 {
2029 struct ace_softc *ace = device_lookup_private(&ace_cd, ACEUNIT(dev));
2030 int part = ACEPART(dev);
2031
2032 DEBUG_PRINT(("aceclose\n"), DEBUG_FUNCS);
2033 if (ace == NULL)
2034 return ENXIO;
2035
2036 mutex_enter(&ace->sc_dk.dk_openlock);
2037
2038 switch (fmt) {
2039 case S_IFCHR:
2040 ace->sc_dk.dk_copenmask &= ~(1 << part);
2041 break;
2042 case S_IFBLK:
2043 ace->sc_dk.dk_bopenmask &= ~(1 << part);
2044 break;
2045 }
2046 ace->sc_dk.dk_openmask =
2047 ace->sc_dk.dk_copenmask | ace->sc_dk.dk_bopenmask;
2048
2049 if (ace->sc_dk.dk_openmask == 0) {
2050
2051 if (!(ace->sc_flags & ACEF_KLABEL))
2052 ace->sc_flags &= ~ACEF_LOADED;
2053
2054 }
2055
2056 mutex_exit(&ace->sc_dk.dk_openlock);
2057 return 0;
2058 }
2059
2060 void
2061 acegetdefaultlabel(struct ace_softc *ace, struct disklabel *lp)
2062 {
2063
2064 DEBUG_PRINT(("acegetdefaultlabel\n"), DEBUG_FUNCS);
2065 memset(lp, 0, sizeof(struct disklabel));
2066
2067 lp->d_secsize = DEV_BSIZE;
2068 lp->d_ntracks = ace->sc_params.CurrentNumberOfHeads;
2069 lp->d_nsectors = ace->sc_params.CurrentSectorsPerTrack;
2070 lp->d_ncylinders = ace->sc_capacity /
2071 (ace->sc_params.CurrentNumberOfHeads *
2072 ace->sc_params.CurrentSectorsPerTrack);
2073 lp->d_secpercyl = lp->d_ntracks * lp->d_nsectors;
2074
2075 lp->d_type = DTYPE_ST506; /* ?!? */
2076
2077 strncpy(lp->d_typename, ace->sc_params.ModelNumber, 16);
2078 strncpy(lp->d_packname, "fictitious", 16);
2079 if (ace->sc_capacity > UINT32_MAX)
2080 lp->d_secperunit = UINT32_MAX;
2081 else
2082 lp->d_secperunit = ace->sc_capacity;
2083 lp->d_rpm = 3600;
2084 lp->d_interleave = 1;
2085 lp->d_flags = 0;
2086
2087 lp->d_partitions[RAW_PART].p_offset = 0;
2088 lp->d_partitions[RAW_PART].p_size =
2089 lp->d_secperunit * (lp->d_secsize / DEV_BSIZE);
2090 lp->d_partitions[RAW_PART].p_fstype = FS_UNUSED;
2091 lp->d_npartitions = RAW_PART + 1;
2092
2093 lp->d_magic = DISKMAGIC;
2094 lp->d_magic2 = DISKMAGIC;
2095 lp->d_checksum = dkcksum(lp);
2096 }
2097
2098 /*
2099 * Fabricate a default disk label, and try to read the correct one.
2100 */
2101 void
2102 acegetdisklabel(struct ace_softc *ace)
2103 {
2104 struct disklabel *lp = ace->sc_dk.dk_label;
2105 const char *errstring;
2106
2107 DEBUG_PRINT(("acegetdisklabel\n"), DEBUG_FUNCS);
2108
2109 memset(ace->sc_dk.dk_cpulabel, 0, sizeof(struct cpu_disklabel));
2110
2111 acegetdefaultlabel(ace, lp);
2112
2113 #ifdef HAS_BAD144_HANDLING
2114 ace->sc_bio.badsect[0] = -1;
2115 #endif
2116
2117 errstring = readdisklabel(MAKEACEDEV(0, device_unit(ace->sc_dev),
2118 RAW_PART), acestrategy, lp,
2119 ace->sc_dk.dk_cpulabel);
2120 if (errstring) {
2121 printf("%s: %s\n", device_xname(ace->sc_dev), errstring);
2122 return;
2123 }
2124
2125 #if DEBUG
2126 if (ACE_DEBUG(DEBUG_WRITES)) {
2127 int i, n = ace->sc_dk.dk_label->d_npartitions;
2128 printf("%s: %d parts\n", device_xname(ace->sc_dev), n);
2129 for (i = 0; i < n; i++) {
2130 printf("\t[%d]: t=%x s=%d o=%d\n", i,
2131 ace->sc_dk.dk_label->d_partitions[i].p_fstype,
2132 ace->sc_dk.dk_label->d_partitions[i].p_size,
2133 ace->sc_dk.dk_label->d_partitions[i].p_offset);
2134 }
2135 }
2136 #endif
2137
2138 #ifdef HAS_BAD144_HANDLING
2139 if ((lp->d_flags & D_BADSECT) != 0)
2140 bad144intern(ace);
2141 #endif
2142 }
2143
2144 void
2145 aceperror(const struct ace_softc *ace)
2146 {
2147 const char *devname = device_xname(ace->sc_dev);
2148 uint32_t Status = ace->sc_bio.r_error;
2149
2150 printf("%s: (", devname);
2151
2152 if (Status == 0)
2153 printf("error not notified");
2154 else
2155 printf("status=x%x", Status);
2156
2157 printf(")\n");
2158 }
2159
2160 int
2161 aceioctl(dev_t dev, u_long xfer, void *addr, int flag, struct lwp *l)
2162 {
2163 struct ace_softc *ace = device_lookup_private(&ace_cd, ACEUNIT(dev));
2164 int error = 0, s;
2165
2166 DEBUG_PRINT(("aceioctl\n"), DEBUG_FUNCS);
2167
2168 if ((ace->sc_flags & ACEF_LOADED) == 0)
2169 return EIO;
2170
2171 error = disk_ioctl(&ace->sc_dk, xfer, addr, flag, l);
2172 if (error != EPASSTHROUGH)
2173 return error;
2174
2175 switch (xfer) {
2176 #ifdef HAS_BAD144_HANDLING
2177 case DIOCSBAD:
2178 if ((flag & FWRITE) == 0)
2179 return EBADF;
2180 ace->sc_dk.dk_cpulabel->bad = *(struct dkbad *)addr;
2181 ace->sc_dk.dk_label->d_flags |= D_BADSECT;
2182 bad144intern(ace);
2183 return 0;
2184 #endif
2185 case DIOCGDINFO:
2186 *(struct disklabel *)addr = *(ace->sc_dk.dk_label);
2187 return 0;
2188
2189 case DIOCGPART:
2190 ((struct partinfo *)addr)->disklab = ace->sc_dk.dk_label;
2191 ((struct partinfo *)addr)->part =
2192 &ace->sc_dk.dk_label->d_partitions[ACEPART(dev)];
2193 return 0;
2194
2195 case DIOCWDINFO:
2196 case DIOCSDINFO:
2197 {
2198 struct disklabel *lp;
2199
2200 if ((flag & FWRITE) == 0)
2201 return EBADF;
2202
2203 lp = (struct disklabel *)addr;
2204
2205 mutex_enter(&ace->sc_dk.dk_openlock);
2206 ace->sc_flags |= ACEF_LABELLING;
2207
2208 error = setdisklabel(ace->sc_dk.dk_label,
2209 lp, /*ace->sc_dk.dk_openmask : */0,
2210 ace->sc_dk.dk_cpulabel);
2211 if (error == 0) {
2212 if (xfer == DIOCWDINFO)
2213 error = writedisklabel(ACELABELDEV(dev),
2214 acestrategy, ace->sc_dk.dk_label,
2215 ace->sc_dk.dk_cpulabel);
2216 }
2217
2218 ace->sc_flags &= ~ACEF_LABELLING;
2219 mutex_exit(&ace->sc_dk.dk_openlock);
2220 return error;
2221 }
2222
2223 case DIOCKLABEL:
2224 if (*(int *)addr)
2225 ace->sc_flags |= ACEF_KLABEL;
2226 else
2227 ace->sc_flags &= ~ACEF_KLABEL;
2228 return 0;
2229
2230 case DIOCWLABEL:
2231 if ((flag & FWRITE) == 0)
2232 return EBADF;
2233 if (*(int *)addr)
2234 ace->sc_flags |= ACEF_WLABEL;
2235 else
2236 ace->sc_flags &= ~ACEF_WLABEL;
2237 return 0;
2238
2239 case DIOCGDEFLABEL:
2240 acegetdefaultlabel(ace, (struct disklabel *)addr);
2241 return 0;
2242
2243 case DIOCCACHESYNC:
2244 return 0;
2245
2246 case DIOCAWEDGE:
2247 {
2248 struct dkwedge_info *dkw = (void *) addr;
2249
2250 if ((flag & FWRITE) == 0)
2251 return EBADF;
2252
2253 /* If the ioctl happens here, the parent is us. */
2254 strcpy(dkw->dkw_parent, device_xname(ace->sc_dev));
2255 return dkwedge_add(dkw);
2256 }
2257
2258 case DIOCDWEDGE:
2259 {
2260 struct dkwedge_info *dkw = (void *) addr;
2261
2262 if ((flag & FWRITE) == 0)
2263 return EBADF;
2264
2265 /* If the ioctl happens here, the parent is us. */
2266 strcpy(dkw->dkw_parent, device_xname(ace->sc_dev));
2267 return dkwedge_del(dkw);
2268 }
2269
2270 case DIOCLWEDGES:
2271 {
2272 struct dkwedge_list *dkwl = (void *) addr;
2273
2274 return dkwedge_list(&ace->sc_dk, dkwl, l);
2275 }
2276
2277 case DIOCGSTRATEGY:
2278 {
2279 struct disk_strategy *dks = (void *)addr;
2280
2281 s = splbio();
2282 strlcpy(dks->dks_name, bufq_getstrategyname(ace->sc_q),
2283 sizeof(dks->dks_name));
2284 splx(s);
2285 dks->dks_paramlen = 0;
2286
2287 return 0;
2288 }
2289
2290 case DIOCSSTRATEGY:
2291 {
2292 struct disk_strategy *dks = (void *)addr;
2293 struct bufq_state *new;
2294 struct bufq_state *old;
2295
2296 if ((flag & FWRITE) == 0) {
2297 return EBADF;
2298 }
2299 if (dks->dks_param != NULL) {
2300 return EINVAL;
2301 }
2302 dks->dks_name[sizeof(dks->dks_name) - 1] = 0; /* ensure term */
2303 error = bufq_alloc(&new, dks->dks_name,
2304 BUFQ_EXACT|BUFQ_SORT_RAWBLOCK);
2305 if (error) {
2306 return error;
2307 }
2308 s = splbio();
2309 old = ace->sc_q;
2310 bufq_move(new, old);
2311 ace->sc_q = new;
2312 splx(s);
2313 bufq_free(old);
2314
2315 return 0;
2316 }
2317
2318 #ifdef USE_ACE_FOR_RECONFIG
2319 /*
2320 * Ok, how do I get this standardized
2321 * [nothing to do with disks either]
2322 */
2323 #define DIOC_FPGA_RECONFIGURE _IOW('d',166, struct ioctl_pt)
2324 case DIOC_FPGA_RECONFIGURE:
2325 {
2326 /*
2327 * BUGBUG This is totally wrong, we need to fault in
2328 * all data in advance.
2329 * Otherwise we get back here with the sysace in a bad state
2330 * (its NOT reentrant!)
2331 */
2332 struct ioctl_pt *pt = (struct ioctl_pt *)addr;
2333 return sysace_send_config(ace,(uint32_t*)pt->data,pt->com);
2334 }
2335 #endif /* USE_ACE_FOR_RECONFIG */
2336
2337 default:
2338 /*
2339 * NB: we get a DIOCGWEDGEINFO, but nobody else handles it
2340 * either
2341 */
2342 DEBUG_PRINT(("aceioctl: unsup x%lx\n", xfer), DEBUG_FUNCS);
2343 return ENOTTY;
2344 }
2345 }
2346
2347 int
2348 acesize(dev_t dev)
2349 {
2350 struct ace_softc *ace;
2351 int part, omask;
2352 int size;
2353
2354 DEBUG_PRINT(("acesize\n"), DEBUG_FUNCS);
2355
2356 ace = device_lookup_private(&ace_cd, ACEUNIT(dev));
2357 if (ace == NULL)
2358 return -1;
2359
2360 part = ACEPART(dev);
2361 omask = ace->sc_dk.dk_openmask & (1 << part);
2362
2363 if (omask == 0 && aceopen(dev, 0, S_IFBLK, NULL) != 0)
2364 return -1;
2365 if (ace->sc_dk.dk_label->d_partitions[part].p_fstype != FS_SWAP)
2366 size = -1;
2367 else
2368 size = ace->sc_dk.dk_label->d_partitions[part].p_size *
2369 (ace->sc_dk.dk_label->d_secsize / DEV_BSIZE);
2370 if (omask == 0 && aceclose(dev, 0, S_IFBLK, NULL) != 0)
2371 return -1;
2372 return size;
2373 }
2374
2375 /* #define ACE_DUMP_NOT_TRUSTED if you just want to watch */
2376 #define ACE_DUMP_NOT_TRUSTED
2377 static int acedoingadump = 0;
2378
2379 /*
2380 * Dump core after a system crash.
2381 */
2382 int
2383 acedump(dev_t dev, daddr_t blkno, void *va, size_t size)
2384 {
2385 struct ace_softc *ace; /* disk unit to do the I/O */
2386 struct disklabel *lp; /* disk's disklabel */
2387 int part, err;
2388 int nblks; /* total number of sectors left to write */
2389
2390 /* Check if recursive dump; if so, punt. */
2391 if (acedoingadump)
2392 return EFAULT;
2393 acedoingadump = 1;
2394
2395 ace = device_lookup_private(&ace_cd, ACEUNIT(dev));
2396 if (ace == NULL)
2397 return ENXIO;
2398
2399 part = ACEPART(dev);
2400
2401 /* Convert to disk sectors. Request must be a multiple of size. */
2402 lp = ace->sc_dk.dk_label;
2403 if ((size % lp->d_secsize) != 0)
2404 return EFAULT;
2405 nblks = size / lp->d_secsize;
2406 blkno = blkno / (lp->d_secsize / DEV_BSIZE);
2407
2408 /* Check transfer bounds against partition size. */
2409 if ((blkno < 0) || ((blkno + nblks) > lp->d_partitions[part].p_size))
2410 return EINVAL;
2411
2412 /* Offset block number to start of partition. */
2413 blkno += lp->d_partitions[part].p_offset;
2414
2415 ace->sc_bp = NULL;
2416 ace->sc_bio.blkno = blkno;
2417 ace->sc_bio.flags = ATA_POLL;
2418 ace->sc_bio.nbytes = nblks * lp->d_secsize;
2419 ace->sc_bio.databuf = va;
2420 #ifndef ACE_DUMP_NOT_TRUSTED
2421 ace->active_xfer = bp;
2422 wakeup(&ace->ch_thread);
2423
2424 switch(ace->sc_bio.error) {
2425 case ETIMEDOUT:
2426 printf("acedump: device timed out");
2427 err = EIO;
2428 break;
2429 case 0:
2430 err = 0;
2431 break;
2432 default:
2433 panic("acedump: unknown error type");
2434 }
2435 if (err != 0) {
2436 printf("\n");
2437 return err;
2438 }
2439 #else /* ACE_DUMP_NOT_TRUSTED */
2440 /* Let's just talk about this first... */
2441 device_printf(ace->sc_dev, ": dump addr 0x%p, size %zu blkno %llx\n",
2442 va, size, blkno);
2443 DELAY(500 * 1000); /* half a second */
2444 err = 0;
2445 #endif
2446
2447 acedoingadump = 0;
2448 return 0;
2449 }
2450
2451 #ifdef HAS_BAD144_HANDLING
2452 /*
2453 * Internalize the bad sector table.
2454 */
2455 void
2456 bad144intern(struct ace_softc *ace)
2457 {
2458 struct dkbad *bt = &ace->sc_dk.dk_cpulabel->bad;
2459 struct disklabel *lp = ace->sc_dk.dk_label;
2460 int i = 0;
2461
2462 DEBUG_PRINT(("bad144intern\n"), DEBUG_XFERS);
2463
2464 for (; i < NBT_BAD; i++) {
2465 if (bt->bt_bad[i].bt_cyl == 0xffff)
2466 break;
2467 ace->sc_bio.badsect[i] =
2468 bt->bt_bad[i].bt_cyl * lp->d_secpercyl +
2469 (bt->bt_bad[i].bt_trksec >> 8) * lp->d_nsectors +
2470 (bt->bt_bad[i].bt_trksec & 0xff);
2471 }
2472 for (; i < NBT_BAD+1; i++)
2473 ace->sc_bio.badsect[i] = -1;
2474 }
2475 #endif
2476
2477 static void
2478 ace_params_to_properties(struct ace_softc *ace)
2479 {
2480 prop_dictionary_t disk_info, odisk_info, geom;
2481 const char *cp;
2482
2483 disk_info = prop_dictionary_create();
2484
2485 cp = ST506;
2486
2487 prop_dictionary_set_cstring_nocopy(disk_info, "type", cp);
2488
2489 geom = prop_dictionary_create();
2490
2491 prop_dictionary_set_uint64(geom, "sectors-per-unit", ace->sc_capacity);
2492
2493 prop_dictionary_set_uint32(geom, "sector-size",
2494 DEV_BSIZE /* XXX 512? */);
2495
2496 prop_dictionary_set_uint16(geom, "sectors-per-track",
2497 ace->sc_params.CurrentSectorsPerTrack);
2498
2499 prop_dictionary_set_uint16(geom, "tracks-per-cylinder",
2500 ace->sc_params.CurrentNumberOfHeads);
2501
2502 prop_dictionary_set_uint64(geom, "cylinders-per-unit",
2503 ace->sc_capacity /
2504 (ace->sc_params.CurrentNumberOfHeads *
2505 ace->sc_params.CurrentSectorsPerTrack));
2506
2507 prop_dictionary_set(disk_info, "geometry", geom);
2508 prop_object_release(geom);
2509
2510 prop_dictionary_set(device_properties(ace->sc_dev),
2511 "disk-info", disk_info);
2512
2513 /*
2514 * Don't release disk_info here; we keep a reference to it.
2515 * disk_detach() will release it when we go away.
2516 */
2517
2518 odisk_info = ace->sc_dk.dk_info;
2519 ace->sc_dk.dk_info = disk_info;
2520 if (odisk_info)
2521 prop_object_release(odisk_info);
2522 }
2523