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 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349
|
/*
NrrdIO: stand-alone code for basic nrrd functionality
Copyright (C) 2008, 2007, 2006, 2005 Gordon Kindlmann
Copyright (C) 2004, 2003, 2002, 2001, 2000, 1999, 1998 University of Utah
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any
damages arising from the use of this software.
Permission is granted to anyone to use this software for any
purpose, including commercial applications, and to alter it and
redistribute it freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must
not claim that you wrote the original software. If you use this
software in a product, an acknowledgment in the product
documentation would be appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must
not be misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
*/
#include "NrrdIO.h"
#include "teemDio.h"
#if TEEM_DIO == 0
#else
/* HEY: these may be SGI-specific */
#include <sys/types.h>
#include <unistd.h>
#include <fcntl.h>
#endif
#if TEEM_DIO == 0
const int airMyDio = 0;
#else
const int airMyDio = 1;
#endif
int airDisableDio = AIR_FALSE;
const char
_airNoDioErr[AIR_NODIO_MAX+2][AIR_STRLEN_SMALL] = {
"(invalid noDio value)",
"CAN TOO do direct I/O!",
"direct I/O apparently not available on this architecture",
"direct I/O apparently not suitable for given file format",
"won't do direct I/O on std{in|out|err}",
"got -1 as file descriptor",
"fcntl(F_DIOINFO) to learn direct I/O specifics failed",
"requested transfer size is too small",
"requested transfer size not a multiple of d_miniosz",
"data memory address not multiple of d_mem",
"current file position not multiple of d_miniosz",
"fcntl(F_SETFL, FDIRECT) to turn on direct I/O failed",
"memalign() test (on a small chuck of memory) failed",
"direct I/O (in air library) has been disabled with airDisableDio"
};
const char *
airNoDioErr(int noDio) {
if (AIR_IN_CL(0, noDio, AIR_NODIO_MAX)) {
return _airNoDioErr[noDio+1];
}
else {
return _airNoDioErr[0];
}
}
/*
******** airDioTest
**
** does everything necessary to assess whether direct IO can be used
** to read a data segment of a given size, from a given file
** descriptor, into a given pointer. The given pointer ptr can be
** NULL, and/or the size can be 0, in order to test the other aspects
** of direct IO. The return value of this is from the airNoDio_* enum.
** Note that airNoDio_okay means, "actually, direct IO *does* seem to
** be possible here".
*/
#if TEEM_DIO == 0
int
airDioTest(int fd, const void *ptr, size_t size) {
AIR_UNUSED(fd);
AIR_UNUSED(ptr);
AIR_UNUSED(size);
/* Teem makefiles think no direct IO is possible on this architecture */
return airNoDio_arch;
}
#else
int
airDioTest(int fd, const void *ptr, size_t size) {
struct dioattr dioinfo;
void *tmp;
int flags;
if (airDisableDio) {
/* user turned direct I/O off */
return airNoDio_disable;
}
if (0 == fd || 1 == fd || 2 == fd) {
/* This was added because I was noticing a problem with piping
between unrrdu programs- sometimes the fread() of the receiving
data through a unix pipe ("|") failed to read all the data. If
the body of this function was bypassed (with "return
airNoDio_disable;", for instance), then the problem went away.
The problematic call seemed to be the fflush() below (Tue Feb 1
06:47:33 EST 2005: which has since been removed with the change
of this function's argument from a FILE * to an integral file
descriptor). I don't think direct I/O is possible on stdin,
stdout, or stdout, since the fcntl() call below fails on stdin
and stdout. However, something about making that fcntl() call
changes something which means that about half the time, the
read() on a piped stdin fails (on an irix6.n32 O2, at
least). So, seems to be safest to just explicitly say that
direct I/O is unavailable, based solely on the file descriptor
number (0, 1, 2). */
return airNoDio_std;
}
if (-1 == fd) {
/* caller probably couldn't get the underlying file descriptor */
return airNoDio_fd;
}
if (0 != fcntl(fd, F_DIOINFO, &dioinfo)) {
/* couldn't learn direct I/O specifics */
return airNoDio_dioinfo;
}
if (size) {
/*
** direct I/O requirements:
** 1) xfer size between d_miniosz and d_maxiosz
** 2) xfer size a multiple of d_miniosz
** 3) memory buffer on d_mem-byte boundary
** 4) file position on d_miniosz-byte boundary
**
** As long as xfer size is >= d_miniosz and meets req. #2, then
** we can break the xfer into d_maxiosz-size pieces of need be.
** We can test #3 here if we're given non-NULL ptr
** We can always test #4
*/
if (size < dioinfo.d_miniosz) {
/* fails req. #1 above */
return airNoDio_small;
}
/* we don't actually check for being too large, since we can always
do IO on d_maxiosz-sized pieces */
if (size % dioinfo.d_miniosz) {
/* fails req. #2 above */
return airNoDio_size;
}
}
if (ptr) {
if ((unsigned long)(ptr) % dioinfo.d_mem) {
/* fails req. #3 above */
return airNoDio_ptr;
}
} else {
tmp = memalign(dioinfo.d_mem, dioinfo.d_miniosz);
if (!tmp) {
/* couldn't even alloc (via memalign) the minimum size */
return airNoDio_test;
}
free(tmp);
}
if (lseek(fd, 0, SEEK_CUR) % dioinfo.d_miniosz) {
/* fails req. #4 above */
return airNoDio_fpos;
}
flags = fcntl(fd, F_GETFL);
if (-1 == fcntl(fd, F_SETFL, flags | FDIRECT)) {
/* couln't turn on direct I/O */
return airNoDio_setfl;
}
/* put things back the way they were */
fcntl(fd, F_SETFL, flags);
/* as far as we know, direct I/O seems workable */
return airNoDio_okay;
}
#endif
/*
******** airDioInfo
**
** does the fcntl stuff to learn the direct IO parameters:
** align: required alignment of memory (pointer must be multiple of this)
** min: minimum size of dio transfer
** max: maximum size of dio transfer
**
** NOTE: this does not try to do any error checking, because it assumes
** that you've already called airDioTest without incident.
*/
#if TEEM_DIO == 0
void
airDioInfo(int *align, int *min, int *max, int fd) {
AIR_UNUSED(align);
AIR_UNUSED(min);
AIR_UNUSED(max);
AIR_UNUSED(fd);
return;
}
#else
void
airDioInfo(int *align, int *min, int *max, int fd) {
struct dioattr dioinfo;
if (align && min && max && !fcntl(fd, F_DIOINFO, &dioinfo)) {
*align = dioinfo.d_mem;
*min = dioinfo.d_miniosz;
*max = dioinfo.d_maxiosz;
}
return;
}
#endif
/*
******** airDioMalloc
**
** does direct IO compatible memory allocation.
**
** NOTE: like airDioInfo, this assumes that you've called airDioTest
** without incident
*/
#if TEEM_DIO == 0
void *
airDioMalloc(size_t size, int fd) {
AIR_UNUSED(size);
AIR_UNUSED(fd);
return NULL;
}
#else
void *
airDioMalloc(size_t size, int fd) {
int align, min, max;
airDioInfo(&align, &min, &max, fd);
return memalign(align, size);
}
#endif
/*
******** airDioRead
**
** like read(), but for direct IO. The idea is that you call this on as
** big a chunk of memory as possible.
**
** NOTE: like airDioInfo, this assumes that you've called airDioTest
** without incident
*/
#if TEEM_DIO == 0
size_t
airDioRead(int fd, void *_ptr, size_t size) {
AIR_UNUSED(fd);
AIR_UNUSED(_ptr);
AIR_UNUSED(size);
return 0;
}
#else
size_t
airDioRead(int fd, void *_ptr, size_t size) {
size_t red, totalred;
int align, min, max, flags;
size_t remain, part;
char *ptr;
if (!( _ptr && airNoDio_okay == airDioTest(fd, _ptr, size) )) {
return 0;
}
flags = fcntl(fd, F_GETFL);
fcntl(fd, F_SETFL, flags | FDIRECT);
airDioInfo(&align, &min, &max, fd);
remain = size;
totalred = 0;
ptr = (char*)_ptr;
do {
part = AIR_MIN(remain, max);
red = read(fd, ptr, part);
totalred += red;
if (red != part) {
break;
}
ptr += red;
remain -= red;
} while (remain);
fcntl(fd, F_SETFL, flags);
return totalred;
}
#endif
/*
******** airDioWrite
**
** like write(), but for direct IO. The idea is that you call this on as
** big a chunk of memory as possible.
**
** NOTE: like airDioInfo, this assumes that you've called airDioTest
** without incident
*/
#if TEEM_DIO == 0
size_t
airDioWrite(int fd, const void *_ptr, size_t size) {
AIR_UNUSED(fd);
AIR_UNUSED(_ptr);
AIR_UNUSED(size);
return 0;
}
#else
size_t
airDioWrite(int fd, const void *_ptr, size_t size) {
size_t rit, totalrit;
int align, min, max, flags;
size_t remain, part;
char *ptr;
if (!( _ptr && (airNoDio_okay == airDioTest(fd, _ptr, size)) )) {
return 0;
}
flags = fcntl(fd, F_GETFL);
fcntl(fd, F_SETFL, flags | FDIRECT);
airDioInfo(&align, &min, &max, fd);
remain = size;
totalrit = 0;
ptr = (char*)_ptr;
do {
part = AIR_MIN(remain, max);
rit = write(fd, ptr, part);
totalrit += rit;
if (rit != part) {
break;
}
ptr += rit;
remain -= rit;
} while (remain);
fcntl(fd, F_SETFL, flags);
return totalrit;
}
#endif
|