-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathreadwrite.c
174 lines (148 loc) · 4.75 KB
/
readwrite.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
/* $cmuPDL: readwrite.c,v 1.3 2010/02/27 11:38:39 rajas Exp $ */
/* $cmuPDL: readwrite.c,v 1.4 2014/01/26 21:16:20 avjaltad Exp $ */
/* readwrite.c
*
* Code to read and write sectors to a "disk" file.
* This is a support file for the "fsck" storage systems laboratory.
*
* author: YOUR NAME HERE
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h> /* for memcpy() */
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <inttypes.h>
#if defined(__FreeBSD__)
#define lseek64 lseek
#endif
/* linux: lseek64 declaration needed here to eliminate compiler warning. */
extern int64_t lseek64(int, int64_t, int);
const unsigned int sector_size_bytes = 512;
static int device; /* disk file descriptor */
/* print_sector: print the contents of a buffer containing one sector.
*
* inputs:
* char *buf: buffer must be >= 512 bytes.
*
* outputs:
* the first 512 bytes of char *buf are printed to stdout.
*
* modifies:
* (none)
*/
void print_sector (unsigned char *buf)
{
int i;
for (i = 0; i < sector_size_bytes; i++) {
printf("%02x", buf[i]);
if (!((i+1) % 32))
printf("\n");
else if (!((i+1) % 4))
printf(" ");
}
printf("%02x", buf[459]);
}
/* read_sectors: read a specified number of sectors into a buffer.
*
* inputs:
* int64 start_sector: the starting sector number to read.
* sector numbering starts with 0.
* int numsectors: the number of sectors to read. must be >= 1.
* int device [GLOBAL]: the disk from which to read.
*
* outputs:
* void *into: the requested number of sectors are copied into here.
*
* modifies:
* void *into
*/
void read_sectors (int64_t start_sector, unsigned int num_sectors, void *into)
{
ssize_t ret;
int64_t lret;
int64_t sector_offset;
ssize_t bytes_to_read;
if (num_sectors == 1) {
printf("Reading sector %"PRId64"\n", start_sector);
} else {
printf("Reading sectors %"PRId64"--%"PRId64"\n",
start_sector, start_sector + (num_sectors - 1));
}
sector_offset = start_sector * sector_size_bytes;
if ((lret = lseek64(device, sector_offset, SEEK_SET)) != sector_offset) {
fprintf(stderr, "Seek to position %"PRId64" failed: "
"returned %"PRId64"\n", sector_offset, lret);
exit(-1);
}
bytes_to_read = sector_size_bytes * num_sectors;
if ((ret = read(device, into, bytes_to_read)) != bytes_to_read) {
fprintf(stderr, "Read sector %"PRId64" length %d failed: "
"returned %"PRId64"\n", start_sector, num_sectors, ret);
exit(-1);
}
}
/* write_sectors: write a buffer into a specified number of sectors.
*
* inputs:
* int64 start_sector: the starting sector number to write.
* sector numbering starts with 0.
* int numsectors: the number of sectors to write. must be >= 1.
* void *from: the requested number of sectors are copied from here.
*
* outputs:
* int device [GLOBAL]: the disk into which to write.
*
* modifies:
* int device [GLOBAL]
*/
void write_sectors (int64_t start_sector, unsigned int num_sectors, void *from)
{
ssize_t ret;
int64_t lret;
int64_t sector_offset;
ssize_t bytes_to_write;
if (num_sectors == 1) {
printf("Reading sector %"PRId64"\n", start_sector);
} else {
printf("Reading sectors %"PRId64"--%"PRId64"\n",
start_sector, start_sector + (num_sectors - 1));
}
sector_offset = start_sector * sector_size_bytes;
if ((lret = lseek64(device, sector_offset, SEEK_SET)) != sector_offset) {
fprintf(stderr, "Seek to position %"PRId64" failed: "
"returned %"PRId64"\n", sector_offset, lret);
exit(-1);
}
bytes_to_write = sector_size_bytes * num_sectors;
if ((ret = write(device, from, bytes_to_write)) != bytes_to_write) {
fprintf(stderr, "Write sector %"PRId64" length %d failed: "
"returned %"PRId64"\n", start_sector, num_sectors, ret);
exit(-1);
}
}
int main (int argc, char **argv)
{
/* This is a sample program. If you want to print out sector 57 of
* the disk, then run the program as:
*
* ./readwrite disk 57
*
* You'll of course want to replace this with your own functions.
*/
unsigned char buf[sector_size_bytes]; /* temporary buffer */
int the_sector; /* IN: sector to read */
if ((device = open(argv[1], O_RDWR)) == -1) {
perror("Could not open device file");
exit(-1);
}
the_sector = atoi(argv[2]);
printf("Dumping sector %d:\n", the_sector);
read_sectors(the_sector, 1, buf);
print_sector(buf);
close(device);
return 0;
}
/* EOF */