fd.c revision 1.9 1 1.9 isaki /* $NetBSD: fd.c,v 1.9 2024/01/07 07:58:34 isaki Exp $ */
2 1.1 minoura
3 1.1 minoura /*
4 1.1 minoura * Copyright (c) 2001 MINOURA Makoto.
5 1.1 minoura * All rights reserved.
6 1.1 minoura *
7 1.1 minoura * Redistribution and use in source and binary forms, with or without
8 1.1 minoura * modification, are permitted provided that the following conditions
9 1.1 minoura * are met:
10 1.1 minoura * 1. Redistributions of source code must retain the above copyright
11 1.1 minoura * notice, this list of conditions and the following disclaimer.
12 1.1 minoura * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 minoura * notice, this list of conditions and the following disclaimer in the
14 1.1 minoura * documentation and/or other materials provided with the distribution.
15 1.1 minoura *
16 1.1 minoura * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17 1.1 minoura * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18 1.1 minoura * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19 1.1 minoura * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20 1.1 minoura * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21 1.1 minoura * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22 1.1 minoura * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23 1.1 minoura * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24 1.1 minoura * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25 1.1 minoura * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 1.1 minoura */
27 1.1 minoura
28 1.1 minoura #include <sys/param.h>
29 1.1 minoura #include <sys/disklabel.h>
30 1.1 minoura #include <lib/libsa/stand.h>
31 1.1 minoura
32 1.5 isaki #include "libx68k.h"
33 1.1 minoura #include "fdvar.h"
34 1.1 minoura #include "iocs.h"
35 1.2 minoura
36 1.5 isaki /* fdopen(struct open_file *f, int id, int part) */
37 1.1 minoura int
38 1.5 isaki fdopen(struct open_file *f, ...)
39 1.1 minoura {
40 1.1 minoura int error;
41 1.1 minoura struct fd_softc *sc;
42 1.1 minoura struct fdfmt fdfmt;
43 1.8 isaki int id;
44 1.8 isaki int part __unused;
45 1.5 isaki va_list ap;
46 1.5 isaki
47 1.5 isaki va_start(ap, f);
48 1.5 isaki id = va_arg(ap, int);
49 1.5 isaki part = va_arg(ap, int);
50 1.5 isaki va_end(ap);
51 1.1 minoura
52 1.1 minoura if (id < 0 || id > 3)
53 1.1 minoura return ENXIO;
54 1.1 minoura sc = alloc(sizeof (struct fd_softc));
55 1.1 minoura
56 1.1 minoura /* lock the medium */
57 1.1 minoura error = IOCS_B_DRVCHK((0x90 + id) << 8, 2);
58 1.1 minoura if ((error & 2) == 0)
59 1.1 minoura return ENXIO;
60 1.1 minoura
61 1.1 minoura /* detect the sector size */
62 1.1 minoura error = IOCS_B_RECALI((0x90 + id) << 8);
63 1.4 isaki error = fd_check_format(id, 0, &sc->fmt);
64 1.1 minoura if (error < 0) {
65 1.1 minoura IOCS_B_DRVCHK((0x90 + id) << 8, 3); /* unlock */
66 1.4 isaki dealloc(sc, sizeof(struct fd_softc));
67 1.1 minoura return -error;
68 1.1 minoura }
69 1.1 minoura
70 1.1 minoura /* check the second side */
71 1.4 isaki error = fd_check_format(id, 1, &fdfmt);
72 1.1 minoura if (error == 0) /* valid second side; set the #heads */
73 1.1 minoura sc->fmt.maxsec.H = fdfmt.maxsec.H;
74 1.1 minoura
75 1.1 minoura sc->unit = id;
76 1.1 minoura f->f_devdata = sc;
77 1.1 minoura
78 1.1 minoura return 0;
79 1.1 minoura }
80 1.1 minoura
81 1.1 minoura int
82 1.4 isaki fdclose(struct open_file *f)
83 1.1 minoura {
84 1.1 minoura struct fd_softc *sc = f->f_devdata;
85 1.1 minoura
86 1.1 minoura IOCS_B_DRVCHK((0x90 + sc->unit) << 8, 3);
87 1.4 isaki dealloc(sc, sizeof(struct fd_softc));
88 1.1 minoura return 0;
89 1.1 minoura }
90 1.1 minoura
91 1.1 minoura int
92 1.4 isaki fdstrategy(void *arg, int rw, daddr_t dblk, size_t size,
93 1.4 isaki void *buf, size_t *rsize)
94 1.1 minoura {
95 1.1 minoura struct fd_softc *sc = arg;
96 1.1 minoura int cyl, head, sect;
97 1.1 minoura int nhead, nsect;
98 1.1 minoura int error, nbytes;
99 1.1 minoura
100 1.1 minoura if (size == 0) {
101 1.1 minoura if (rsize)
102 1.1 minoura *rsize = 0;
103 1.1 minoura return 0;
104 1.1 minoura }
105 1.4 isaki nbytes = howmany(size, 128 << sc->fmt.minsec.N)
106 1.1 minoura * (128 << sc->fmt.minsec.N);
107 1.1 minoura
108 1.1 minoura nhead = sc->fmt.maxsec.H - sc->fmt.minsec.H + 1;
109 1.1 minoura nsect = sc->fmt.maxsec.R - sc->fmt.minsec.R + 1;
110 1.1 minoura
111 1.9 isaki sect = dblk % nsect + sc->fmt.minsec.R;
112 1.1 minoura head = (dblk / nsect) % nhead + sc->fmt.minsec.H;
113 1.1 minoura cyl = (dblk / nsect) / nhead + sc->fmt.minsec.C;
114 1.1 minoura
115 1.6 tsutsui error = IOCS_B_READ((sc->unit + 0x90) * 256 + 0x70,
116 1.1 minoura ((sc->fmt.minsec.N << 24) |
117 1.1 minoura (cyl << 16) |
118 1.1 minoura (head << 8) |
119 1.1 minoura (sect)),
120 1.1 minoura nbytes,
121 1.1 minoura buf);
122 1.1 minoura if (error & 0xf8ffff00) {
123 1.1 minoura nbytes = 0;
124 1.1 minoura error = EIO;
125 1.4 isaki } else {
126 1.1 minoura error = 0;
127 1.4 isaki }
128 1.1 minoura
129 1.1 minoura if (rsize)
130 1.1 minoura *rsize = nbytes;
131 1.1 minoura return error;
132 1.1 minoura }
133