Changeset 19562 in vbox for trunk/src/VBox/Runtime/r3/freebsd
- Timestamp:
- May 10, 2009 9:44:16 PM (16 years ago)
- svn:sync-xref-src-repo-rev:
- 47117
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Runtime/r3/freebsd/fileaio-freebsd.cpp
r19187 r19562 47 47 #include <sys/event.h> 48 48 #include <sys/time.h> 49 #include <sys/sysctl.h> 49 50 #include <aio.h> 50 51 #include <errno.h> … … 84 85 * element. */ 85 86 struct aiocb AioCB; 87 /** Current state the request is in. */ 88 RTFILEAIOREQSTATE enmState; 86 89 /** Flag whether this is a flush request. */ 87 90 bool fFlush; … … 107 110 #define AIO_MAXIMUM_REQUESTS_PER_CONTEXT 64 108 111 112 RTR3DECL(int) RTFileAioGetLimits(PRTFILEAIOLIMITS pAioLimits) 113 { 114 int rcBSD = 0; 115 AssertPtrReturn(pAioLimits, VERR_INVALID_POINTER); 116 117 /* 118 * The AIO API is implemented in a kernel module which is not 119 * loaded by default. 120 * If it is loaded there are additional sysctl parameters. 121 */ 122 int cReqsOutstandingMax = 0; 123 size_t cbParameter = sizeof(int); 124 125 rcBSD = sysctlbyname("vfs.aio.max_aio_per_proc", /* name */ 126 &cReqsOutstandingMax, /* Where to store the old value. */ 127 &cbParameter, /* Size of the memory pointed to. */ 128 NULL, /* Where the new value is located. */ 129 NULL); /* Where the size of the new value is stored. */ 130 if (rcBSD == -1) 131 { 132 /* ENOENT means the value is unknown thus the module is not loaded. */ 133 if (errno == ENOENT) 134 return VERR_NOT_SUPPORTED; 135 else 136 return RTErrConvertFromErrno(errno); 137 } 138 139 pAioLimits->cReqsOutstandingMax = cReqsOutstandingMax; 140 pAioLimits->cbBufferAlignment = 0; 141 142 return VINF_SUCCESS; 143 } 144 109 145 RTR3DECL(int) RTFileAioReqCreate(PRTFILEAIOREQ phReq) 110 146 { … … 120 156 pReqInt->pCtxInt = NULL; 121 157 pReqInt->u32Magic = RTFILEAIOREQ_MAGIC; 158 RTFILEAIOREQ_SET_STATE(pReqInt, COMPLETED); 122 159 123 160 *phReq = (RTFILEAIOREQ)pReqInt; … … 126 163 } 127 164 128 RTDECL( void) RTFileAioReqDestroy(RTFILEAIOREQ hReq)165 RTDECL(int) RTFileAioReqDestroy(RTFILEAIOREQ hReq) 129 166 { 130 167 /* … … 132 169 */ 133 170 if (hReq == NIL_RTFILEAIOREQ) 134 return ;171 return VINF_SUCCESS; 135 172 PRTFILEAIOREQINTERNAL pReqInt = hReq; 136 RTFILEAIOREQ_VALID_RETURN_VOID(pReqInt); 173 RTFILEAIOREQ_VALID_RETURN(pReqInt); 174 RTFILEAIOREQ_NOT_STATE_RETURN_RC(pReqInt, SUBMITTED, VERR_FILE_AIO_IN_PROGRESS); 137 175 138 176 /* … … 141 179 ASMAtomicUoWriteU32(&pReqInt->u32Magic, ~RTFILEAIOREQ_MAGIC); 142 180 RTMemFree(pReqInt); 181 return VINF_SUCCESS; 143 182 } 144 183 … … 156 195 PRTFILEAIOREQINTERNAL pReqInt = hReq; 157 196 RTFILEAIOREQ_VALID_RETURN(pReqInt); 197 RTFILEAIOREQ_NOT_STATE_RETURN_RC(pReqInt, SUBMITTED, VERR_FILE_AIO_IN_PROGRESS); 158 198 Assert(hFile != NIL_RTFILE); 159 199 AssertPtr(pvBuf); … … 171 211 pReqInt->pCtxInt = NULL; 172 212 pReqInt->Rc = VERR_FILE_AIO_IN_PROGRESS; 213 RTFILEAIOREQ_SET_STATE(pReqInt, PREPARED); 173 214 174 215 return VINF_SUCCESS; … … 195 236 RTFILEAIOREQ_VALID_RETURN(pReqInt); 196 237 Assert(hFile != NIL_RTFILE); 238 RTFILEAIOREQ_NOT_STATE_RETURN_RC(pReqInt, SUBMITTED, VERR_FILE_AIO_IN_PROGRESS); 197 239 198 240 pReqInt->fFlush = true; 199 241 pReqInt->AioCB.aio_fildes = (int)hFile; 200 242 pReqInt->pvUser = pvUser; 243 RTFILEAIOREQ_SET_STATE(pReqInt, PREPARED); 201 244 202 245 return VINF_SUCCESS; … … 215 258 PRTFILEAIOREQINTERNAL pReqInt = hReq; 216 259 RTFILEAIOREQ_VALID_RETURN(pReqInt); 260 RTFILEAIOREQ_STATE_RETURN_RC(pReqInt, SUBMITTED, VERR_FILE_AIO_NOT_SUBMITTED); 261 217 262 218 263 int rcBSD = aio_cancel(pReqInt->AioCB.aio_fildes, &pReqInt->AioCB); … … 228 273 229 274 ASMAtomicDecS32(&pReqInt->pCtxInt->cRequests); 275 pReqInt->Rc = VERR_FILE_AIO_CANCELED; 276 RTFILEAIOREQ_SET_STATE(pReqInt, COMPLETED); 230 277 return VINF_SUCCESS; 231 278 } … … 243 290 RTFILEAIOREQ_VALID_RETURN(pReqInt); 244 291 AssertPtrNull(pcbTransfered); 245 246 if ( (pReqInt->Rc != VERR_FILE_AIO_IN_PROGRESS) 292 RTFILEAIOREQ_NOT_STATE_RETURN_RC(pReqInt, SUBMITTED, VERR_FILE_AIO_IN_PROGRESS); 293 RTFILEAIOREQ_NOT_STATE_RETURN_RC(pReqInt, PREPARED, VERR_FILE_AIO_NOT_SUBMITTED); 294 295 if ( (RT_SUCCESS(pReqInt->Rc)) 247 296 && (pcbTransfered)) 248 297 *pcbTransfered = pReqInt->cbTransfered; … … 306 355 } 307 356 308 RTDECL(int) RTFileAioCtxSubmit(RTFILEAIOCTX hAioCtx, PRTFILEAIOREQ pahReqs, size_t cReqs , size_t *pcReqs)357 RTDECL(int) RTFileAioCtxSubmit(RTFILEAIOCTX hAioCtx, PRTFILEAIOREQ pahReqs, size_t cReqs) 309 358 { 310 359 /* 311 360 * Parameter validation. 312 361 */ 313 AssertPtrReturn(pcReqs, VERR_INVALID_POINTER); 314 *pcReqs = 0; 362 int rc = VINF_SUCCESS; 315 363 PRTFILEAIOCTXINTERNAL pCtxInt = hAioCtx; 316 364 RTFILEAIOCTX_VALID_RETURN(pCtxInt); … … 330 378 { 331 379 pReqInt = pahReqs[i]; 332 RTFILEAIOREQ_VALID_RETURN(pReqInt); 380 if (RTFILEAIOREQ_IS_NOT_VALID(pReqInt)) 381 { 382 /* Undo everything and stop submitting. */ 383 for (size_t iUndo = 0; iUndo < i; iUndo++) 384 { 385 pReqInt = pahReqs[iUndo]; 386 RTFILEAIOREQ_SET_STATE(pReqInt, PREPARED); 387 pReqInt->pCtxInt = NULL; 388 pReqInt->AioCB.aio_sigevent.sigev_notify_kqueue = 0; 389 } 390 rc = VERR_INVALID_HANDLE; 391 break; 392 } 333 393 334 394 pReqInt->AioCB.aio_sigevent.sigev_notify_kqueue = pCtxInt->iKQueue; 335 395 pReqInt->pCtxInt = pCtxInt; 396 RTFILEAIOREQ_SET_STATE(pReqInt, SUBMITTED); 336 397 337 398 if (pReqInt->fFlush) … … 346 407 rcBSD = lio_listio(LIO_NOWAIT, (struct aiocb **)pahReqs, cReqsSubmit, NULL); 347 408 if (RT_UNLIKELY(rcBSD < 0)) 348 return RTErrConvertFromErrno(errno); 409 { 410 if (rcBSD == EAGAIN) 411 rc = VERR_FILE_AIO_INSUFFICIENT_RESSOURCES; 412 else 413 rc = RTErrConvertFromErrno(errno); 414 415 /* Check which requests got actually submitted and which not. */ 416 for (i = 0; i < cReqs; i++) 417 { 418 pReqInt = pahReqs[i]; 419 rcBSD = aio_error(&pReqInt->AioCB); 420 if (rcBSD == EINVAL) 421 { 422 /* Was not submitted. */ 423 RTFILEAIOREQ_SET_STATE(pReqInt, PREPARED); 424 pReqInt->pCtxInt = NULL; 425 } 426 else if (rcBSD != EINPROGRESS) 427 { 428 /* The request encountered an error. */ 429 RTFILEAIOREQ_SET_STATE(pReqInt, COMPLETED); 430 pReqInt->Rc = RTErrConvertFromErrno(rcBSD); 431 pReqInt->pCtxInt = NULL; 432 pReqInt->cbTransfered = 0; 433 } 434 } 435 break; 436 } 349 437 350 438 ASMAtomicAddS32(&pCtxInt->cRequests, cReqsSubmit); 351 439 cReqs -= cReqsSubmit; 352 440 pahReqs += cReqsSubmit; 353 *pcReqs += cReqsSubmit;354 441 } 355 442 … … 368 455 rcBSD = aio_fsync(O_SYNC, &pReqInt->AioCB); 369 456 if (RT_UNLIKELY(rcBSD < 0)) 370 return RTErrConvertFromErrno(errno); 457 { 458 RTFILEAIOREQ_SET_STATE(pReqInt, COMPLETED); 459 pReqInt->Rc = RTErrConvertFromErrno(errno); 460 pReqInt->cbTransfered = 0; 461 return pReqInt->Rc; 462 } 371 463 372 464 ASMAtomicIncS32(&pCtxInt->cRequests); 373 465 cReqs--; 374 466 pahReqs++; 375 *pcReqs++;376 467 } 377 468 } 378 469 } while (cReqs); 379 470 380 return VINF_SUCCESS;471 return rc; 381 472 } 382 473 … … 447 538 { 448 539 PRTFILEAIOREQINTERNAL pReqInt = (PRTFILEAIOREQINTERNAL)aKEvents[i].udata; 540 AssertPtr(pReqInt); 541 Assert(pReqInt->u32Magic == RTFILEAIOREQ_MAGIC); 449 542 450 543 /* … … 468 561 pReqInt->cbTransfered = cbTransfered; 469 562 } 563 RTFILEAIOREQ_SET_STATE(pReqInt, COMPLETED); 470 564 pahReqs[cRequestsCompleted++] = (RTFILEAIOREQ)pReqInt; 471 565 }
Note:
See TracChangeset
for help on using the changeset viewer.