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
|
/*
* getsectsize.c --- get the sector size of a device.
*
* Copyright (C) 1995, 1995 Theodore Ts'o.
* Copyright (C) 2003 VMware, Inc.
*
* %Begin-Header%
* This file may be redistributed under the terms of the GNU Library
* General Public License, version 2.
* %End-Header%
*/
#ifndef _LARGEFILE_SOURCE
#define _LARGEFILE_SOURCE
#endif
#ifndef _LARGEFILE64_SOURCE
#define _LARGEFILE64_SOURCE
#endif
#include "config.h"
#include <stdio.h>
#if HAVE_UNISTD_H
#include <unistd.h>
#endif
#if HAVE_ERRNO_H
#include <errno.h>
#endif
#include <fcntl.h>
#ifdef HAVE_SYS_DISK_H
#include <sys/disk.h>
#endif
#ifdef HAVE_LINUX_FD_H
#include <sys/ioctl.h>
#include <linux/fd.h>
#endif
#if defined(__linux__) && defined(_IO)
#if !defined(BLKSSZGET)
#define BLKSSZGET _IO(0x12,104)/* get block device sector size */
#endif
#if !defined(BLKPBSZGET)
#define BLKPBSZGET _IO(0x12,123)/* get block physical sector size */
#endif
#endif
#include "ext2_fs.h"
#include "ext2fs.h"
/*
* Returns the logical sector size of a device
*/
errcode_t ext2fs_get_device_sectsize(const char *file, int *sectsize)
{
#ifdef _WIN32
*sectsize = 512; // just guessing
return 0;
#else // not _WIN32
int fd;
fd = ext2fs_open_file(file, O_RDONLY, 0);
if (fd < 0)
return errno;
#ifdef BLKSSZGET
if (ioctl(fd, BLKSSZGET, sectsize) >= 0) {
close(fd);
return 0;
}
#endif
#ifdef DIOCGSECTORSIZE
if (ioctl(fd, DIOCGSECTORSIZE, sectsize) >= 0) {
close(fd);
return 0;
}
#endif
*sectsize = 0;
close(fd);
return 0;
#endif // ifdef _WIN32
}
/*
* Return desired alignment for direct I/O
*/
int ext2fs_get_dio_alignment(int fd)
{
int align = 0;
#ifdef BLKSSZGET
if (ioctl(fd, BLKSSZGET, &align) < 0)
align = 0;
#endif
#ifdef DIOCGSECTORSIZE
if (align <= 0 &&
ioctl(fd, DIOCGSECTORSIZE, &align) < 0)
align = 0;
#endif
#ifdef _SC_PAGESIZE
if (align <= 0)
align = sysconf(_SC_PAGESIZE);
#endif
#ifdef HAVE_GETPAGESIZE
if (align <= 0)
align = getpagesize();
#endif
if (align <= 0)
align = 4096;
return align;
}
/*
* Returns the physical sector size of a device
*/
errcode_t ext2fs_get_device_phys_sectsize(const char *file, int *sectsize)
{
#ifdef _WIN32
return ext2fs_get_device_sectsize(file, sectsize);
#else // not _WIN32
int fd;
fd = ext2fs_open_file(file, O_RDONLY, 0);
if (fd < 0)
return errno;
#ifdef BLKPBSZGET
if (ioctl(fd, BLKPBSZGET, sectsize) >= 0) {
close(fd);
return 0;
}
#endif
#ifdef DIOCGSECTORSIZE
/* This isn't really the physical sector size, but FreeBSD
* doesn't seem to have this concept. */
if (ioctl(fd, DIOCGSECTORSIZE, sectsize) >= 0) {
close(fd);
return 0;
}
#endif
*sectsize = 0;
close(fd);
return 0;
#endif // ifdef _WIN32
}
|