]> git.lizzy.rs Git - plan9front.git/blob - sys/src/cmd/cwfs/wren.c
merge
[plan9front.git] / sys / src / cmd / cwfs / wren.c
1 /*
2  * drive disks
3  * used to be just scsi disks, and issued scsi commands directly to the host
4  * adapter, but now it just does normal i/o.
5  */
6 #include "all.h"
7
8 enum { Sectorsz = 512, };               /* usual disk sector size */
9
10 typedef struct  Wren    Wren;
11 struct  Wren
12 {
13         long    block;                  /* size of a block -- from config */
14         Devsize nblock;                 /* number of blocks -- from config */
15         long    mult;                   /* multiplier to get physical blocks */
16         Devsize max;                    /* number of logical blocks */
17 };
18
19 char *
20 dataof(char *file)
21 {
22         char *datanm;
23         Dir *dir;
24
25         dir = dirstat(file);
26         if (dir != nil && dir->mode & DMDIR)
27                 datanm = smprint("%s/data", file);
28         else
29                 datanm = strdup(file);
30         free(dir);
31         return datanm;
32 }
33
34 void
35 wreninit(Device *d)
36 {
37         Wren *dr;
38         Dir *dir;
39
40         if(d->private)
41                 return;
42         d->private = dr = ialloc(sizeof(Wren), 0);
43
44         if (d->wren.file)
45                 d->wren.sddata = dataof(d->wren.file);
46         else {
47                 d->wren.sddir = sdof(d);
48                 d->wren.sddata = smprint("%s/data", d->wren.sddir);
49         }
50
51         d->wren.fd = open(d->wren.sddata, ORDWR);
52         if (d->wren.fd < 0)
53                 panic("wreninit: can't open %s for %Z: %r", d->wren.sddata, d);
54
55         dr->block = inqsize(d->wren.sddata);
56         if(dr->block <= 0 || dr->block >= 16*1024) {
57                 if(chatty)
58                         print("\twreninit %Z block size %ld, setting to %d\n",
59                                 d, dr->block, Sectorsz);
60                 dr->block = Sectorsz;
61         }
62
63         dir = dirfstat(d->wren.fd);
64         dr->nblock = dir->length / dr->block;
65         free(dir);
66
67         dr->mult = (RBUFSIZE + dr->block - 1) / dr->block;
68         dr->max = (dr->nblock + 1) / dr->mult;
69         if(chatty){
70                 print("\tdisk drive %Z: %,lld %ld-byte sectors, ",
71                         d, (Wideoff)dr->nblock, dr->block);
72                 print("%,lld %d-byte blocks\n", (Wideoff)dr->max, RBUFSIZE);
73                 print("\t\t%ld multiplier\n", dr->mult);
74         }
75 }
76
77 Devsize
78 wrensize(Device *d)
79 {
80         return ((Wren *)d->private)->max;
81 }
82
83 int
84 wrenread(Device *d, Off b, void *c)
85 {
86         int r = 0;
87         Wren *dr = d->private;
88
89         if (dr == nil)
90                 panic("wrenread: no drive (%Z) block %lld", d, (Wideoff)b);
91         if(b >= dr->max) {
92                 fprint(2, "wrenread: block out of range %Z(%lld)\n", d, (Wideoff)b);
93                 r = 0x040;
94         } else if (pread(d->wren.fd, c, RBUFSIZE, (vlong)b*RBUFSIZE) != RBUFSIZE) {
95                 fprint(2, "wrenread: error on %Z(%lld): %r\n", d, (Wideoff)b);
96                 cons.nwrenre++;
97                 r = 1;
98         }
99         return r;
100 }
101
102 int
103 wrenwrite(Device *d, Off b, void *c)
104 {
105         int r = 0;
106         Wren *dr = d->private;
107
108         if (dr == nil)
109                 panic("wrenwrite: no drive (%Z) block %lld", d, (Wideoff)b);
110         if(b >= dr->max) {
111                 fprint(2, "wrenwrite: block out of range %Z(%lld)\n",
112                         d, (Wideoff)b);
113                 r = 0x040;
114         } else if (pwrite(d->wren.fd, c, RBUFSIZE, (vlong)b*RBUFSIZE) != RBUFSIZE) {
115                 fprint(2, "wrenwrite: error on %Z(%lld): %r\n", d, (Wideoff)b);
116                 cons.nwrenwe++;
117                 r = 1;
118         }
119
120         return r;
121 }