xref: /freebsd/usr.sbin/fdwrite/fdwrite.c (revision 05c7a37afb48ddd5ee1bd921a5d46fe59cc70b15)
1 /*
2  * ----------------------------------------------------------------------------
3  * "THE BEER-WARE LICENSE" (Revision 42):
4  * <phk@login.dkuug.dk> wrote this file.  As long as you retain this notice you
5  * can do whatever you want with this stuff. If we meet some day, and you think
6  * this stuff is worth it, you can buy me a beer in return.   Poul-Henning Kamp
7  * ----------------------------------------------------------------------------
8  *
9  * $Id: fdwrite.c,v 1.2 1994/10/14 16:03:33 joerg Exp $
10  *
11  */
12 
13 #include <stdio.h>
14 #include <stdlib.h>
15 #include <unistd.h>
16 #include <fcntl.h>
17 #include <strings.h>
18 #include <ctype.h>
19 
20 #include <errno.h>
21 #include <machine/ioctl_fd.h>
22 
23 int
24 format_track(int fd, int cyl, int secs, int head, int rate,
25      int gaplen, int secsize, int fill,int interleave)
26 {
27     struct fd_formb f;
28     register int i,j;
29     int il[100];
30 
31     memset(il,0,sizeof il);
32     for(j = 0, i = 1; i <= secs; i++) {
33 	while(il[(j%secs)+1]) j++;
34 	il[(j%secs)+1] = i;
35 	j += interleave;
36     }
37 
38     f.format_version = FD_FORMAT_VERSION;
39     f.head = head;
40     f.cyl = cyl;
41     f.transfer_rate = rate;
42 
43     f.fd_formb_secshift = secsize;
44     f.fd_formb_nsecs = secs;
45     f.fd_formb_gaplen = gaplen;
46     f.fd_formb_fillbyte = fill;
47     for(i = 0; i < secs; i++) {
48 	f.fd_formb_cylno(i) = cyl;
49 	f.fd_formb_headno(i) = head;
50 	f.fd_formb_secno(i) = il[i+1];
51 	f.fd_formb_secsize(i) = secsize;
52     }
53     return ioctl(fd, FD_FORM, (caddr_t)&f);
54 }
55 
56 static void
57 usage ()
58 {
59 	printf("Usage:\n\tfdwrite [-v] [-f inputfile] [-d device]\n");
60 	exit(2);
61 }
62 
63 int
64 main(int argc, char **argv)
65 {
66     int inputfd = -1, c, fdn = 0, i,j,fd;
67     int bpt, verbose=1, nbytes=0, track;
68     char *device= "/dev/rfd0", *trackbuf = 0,*vrfybuf = 0;
69     struct fd_type fdt;
70     FILE *tty;
71 
72     setbuf(stdout,0);
73     while((c = getopt(argc, argv, "d:s:f:v")) != -1)
74 	    switch(c) {
75 	    case 'd':	/* Which drive */
76 		    device = optarg;
77 		    break;
78 
79 	    case 'f':	/* input file */
80 		    if (inputfd >= 0)
81 			    close(inputfd);
82 		    inputfd = open(optarg,O_RDONLY);
83 		    if (inputfd < 0) {
84 			    perror(optarg);
85 			    exit(1);
86 		    }
87 		    break;
88 
89 	    case 'v':  /* Toggle verbosity */
90 		    verbose = !verbose;
91 		    break;
92 
93 	    case '?': default:
94 		    usage();
95 	    }
96 
97     if (inputfd < 0)
98 	inputfd = 0;
99 
100     if(optind < argc)
101 	    usage();
102 
103     tty = fopen("/dev/tty","r+");
104     if(!tty) {
105 	    perror("/dev/tty");
106 	    exit(1);
107     }
108     setbuf(tty,0);
109 
110     for(j=1;j > 0;) {
111         fdn++;
112 	fprintf(tty,
113 	    "Please insert floppy #%d in drive %s and press return >",
114 		fdn,device);
115 	while(1) {
116 	    i = getc(tty);
117 	    if(i == '\n') break;
118 	}
119 
120 	if((fd = open(device, O_RDWR)) < 0) {
121 	    perror(device);
122 	    exit(1);
123 	}
124 
125 	if(ioctl(fd, FD_GTYPE, &fdt) < 0) {
126 	    fprintf(stderr, "fdformat: not a floppy disk: %s\n", device);
127 	    exit(1);
128 	}
129 
130 	bpt = fdt.sectrac * (1<<fdt.secsize) * 128;
131 	if(!trackbuf) {
132 	    trackbuf = malloc(bpt);
133 	    if(!trackbuf) { perror("malloc"); exit(1); }
134 	}
135 	if(!vrfybuf) {
136 	    vrfybuf = malloc(bpt);
137 	    if(!vrfybuf) { perror("malloc"); exit(1); }
138 	}
139 
140 	if(fdn == 1) {
141 	    if(verbose) {
142 		printf("Format: %d cylinders, %d heads, %d sectors, %d bytes = %dkb\n",
143 		fdt.tracks,fdt.heads,fdt.sectrac,(1<<fdt.secsize) * 128,
144 		fdt.tracks*bpt*fdt.heads/1024);
145 
146 	    }
147 	    memset(trackbuf,0,bpt);
148 	    for(j=0;inputfd >= 0 && j<bpt;j+=i) {
149 		if(!(i = read(inputfd,trackbuf+j,bpt-j))) {
150 		    close(inputfd);
151 		    inputfd = -1;
152 		    break;
153 		}
154 		nbytes += i;
155 	    }
156 	}
157 	for (track = 0; track < fdt.tracks * fdt.heads; track++) {
158 	    if(verbose) printf("\r%3d ",fdt.tracks * fdt.heads-track);
159 	    if(verbose) putc((j ? 'I':'Z'),stdout);
160 	    format_track(fd, track / fdt.heads, fdt.sectrac, track % fdt.heads,
161 		    fdt.trans, fdt.f_gap, fdt.secsize, 0xe6,
162 		    fdt.f_inter);
163 	    if(verbose) putc('F',stdout);
164 
165 	    if (lseek (fd, (long) track*bpt, 0) < 0) {
166 		perror("lseek");
167 		exit (1);
168 	    }
169 	    if (write (fd, trackbuf, bpt) != bpt) {
170 		perror("write");
171 		exit (1);
172 	    }
173 	    if(verbose) putc('W',stdout);
174 
175 	    if (lseek (fd, (long) track*bpt, 0) < 0) {
176 		perror("lseek");
177 		exit (1);
178 	    }
179 	    if (read (fd, vrfybuf, bpt) != bpt) {
180 		perror("read");
181 		exit (1);
182 	    }
183 	    if(verbose) putc('R',stdout);
184 
185 	    if (memcmp(trackbuf,vrfybuf,bpt)) {
186 		perror("compare");
187 		exit (1);
188 	    }
189 	    if(verbose) putc('C',stdout);
190 
191 	    memset(trackbuf,0,bpt);
192 	    for(j=0;inputfd >= 0 && j<bpt;j+=i) {
193 		if(!(i = read(inputfd,trackbuf+j,bpt-j))) {
194 		    close(inputfd);
195 		    inputfd = -1;
196 		    break;
197 		}
198 		nbytes += i;
199 	    }
200 	}
201 	close(fd);
202 	putc('\r',stdout);
203     }
204     if(verbose)
205 	printf("%d bytes on %d flopp%s\n",nbytes,fdn,fdn==1?"y":"ies");
206     exit(0);
207 }
208