VirtualBox

Ignore:
Timestamp:
May 10, 2009 9:44:16 PM (16 years ago)
Author:
vboxsync
svn:sync-xref-src-repo-rev:
47117
Message:

Runtime/Aio: Change API again

  • pcReqs in RTFileAioCtxSubmit is useless because it does not specify which request fails. Removed it again and made it possible to get the state of a request through RTFileAioReqGetRC()
  • Introduce request states for the first point and to catch more errors using the API before a system dependent call is made to return the same error codes one every system for the same cause.
  • Add RTFileAioGetLimits to get global limits and indication for AIO support.
  • General cleanups and fixes
File:
1 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/Runtime/r3/freebsd/fileaio-freebsd.cpp

    r19187 r19562  
    4747#include <sys/event.h>
    4848#include <sys/time.h>
     49#include <sys/sysctl.h>
    4950#include <aio.h>
    5051#include <errno.h>
     
    8485     *  element. */
    8586    struct aiocb           AioCB;
     87    /** Current state the request is in. */
     88    RTFILEAIOREQSTATE      enmState;
    8689    /** Flag whether this is a flush request. */
    8790    bool                   fFlush;
     
    107110#define AIO_MAXIMUM_REQUESTS_PER_CONTEXT 64
    108111
     112RTR3DECL(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
    109145RTR3DECL(int) RTFileAioReqCreate(PRTFILEAIOREQ phReq)
    110146{
     
    120156    pReqInt->pCtxInt                   = NULL;
    121157    pReqInt->u32Magic                  = RTFILEAIOREQ_MAGIC;
     158    RTFILEAIOREQ_SET_STATE(pReqInt, COMPLETED);
    122159
    123160    *phReq = (RTFILEAIOREQ)pReqInt;
     
    126163}
    127164
    128 RTDECL(void) RTFileAioReqDestroy(RTFILEAIOREQ hReq)
     165RTDECL(int) RTFileAioReqDestroy(RTFILEAIOREQ hReq)
    129166{
    130167    /*
     
    132169     */
    133170    if (hReq == NIL_RTFILEAIOREQ)
    134         return;
     171        return VINF_SUCCESS;
    135172    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);
    137175
    138176    /*
     
    141179    ASMAtomicUoWriteU32(&pReqInt->u32Magic, ~RTFILEAIOREQ_MAGIC);
    142180    RTMemFree(pReqInt);
     181    return VINF_SUCCESS;
    143182}
    144183
     
    156195    PRTFILEAIOREQINTERNAL pReqInt = hReq;
    157196    RTFILEAIOREQ_VALID_RETURN(pReqInt);
     197    RTFILEAIOREQ_NOT_STATE_RETURN_RC(pReqInt, SUBMITTED, VERR_FILE_AIO_IN_PROGRESS);
    158198    Assert(hFile != NIL_RTFILE);
    159199    AssertPtr(pvBuf);
     
    171211    pReqInt->pCtxInt              = NULL;
    172212    pReqInt->Rc                   = VERR_FILE_AIO_IN_PROGRESS;
     213    RTFILEAIOREQ_SET_STATE(pReqInt, PREPARED);
    173214
    174215    return VINF_SUCCESS;
     
    195236    RTFILEAIOREQ_VALID_RETURN(pReqInt);
    196237    Assert(hFile != NIL_RTFILE);
     238    RTFILEAIOREQ_NOT_STATE_RETURN_RC(pReqInt, SUBMITTED, VERR_FILE_AIO_IN_PROGRESS);
    197239
    198240    pReqInt->fFlush           = true;
    199241    pReqInt->AioCB.aio_fildes = (int)hFile;
    200242    pReqInt->pvUser           = pvUser;
     243    RTFILEAIOREQ_SET_STATE(pReqInt, PREPARED);
    201244
    202245    return VINF_SUCCESS;
     
    215258    PRTFILEAIOREQINTERNAL pReqInt = hReq;
    216259    RTFILEAIOREQ_VALID_RETURN(pReqInt);
     260    RTFILEAIOREQ_STATE_RETURN_RC(pReqInt, SUBMITTED, VERR_FILE_AIO_NOT_SUBMITTED);
     261
    217262
    218263    int rcBSD = aio_cancel(pReqInt->AioCB.aio_fildes, &pReqInt->AioCB);
     
    228273
    229274        ASMAtomicDecS32(&pReqInt->pCtxInt->cRequests);
     275        pReqInt->Rc = VERR_FILE_AIO_CANCELED;
     276        RTFILEAIOREQ_SET_STATE(pReqInt, COMPLETED);
    230277        return VINF_SUCCESS;
    231278    }
     
    243290    RTFILEAIOREQ_VALID_RETURN(pReqInt);
    244291    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))
    247296        && (pcbTransfered))
    248297        *pcbTransfered = pReqInt->cbTransfered;
     
    306355}
    307356
    308 RTDECL(int) RTFileAioCtxSubmit(RTFILEAIOCTX hAioCtx, PRTFILEAIOREQ pahReqs, size_t cReqs, size_t *pcReqs)
     357RTDECL(int) RTFileAioCtxSubmit(RTFILEAIOCTX hAioCtx, PRTFILEAIOREQ pahReqs, size_t cReqs)
    309358{
    310359    /*
    311360     * Parameter validation.
    312361     */
    313     AssertPtrReturn(pcReqs, VERR_INVALID_POINTER);
    314     *pcReqs = 0;
     362    int rc = VINF_SUCCESS;
    315363    PRTFILEAIOCTXINTERNAL pCtxInt = hAioCtx;
    316364    RTFILEAIOCTX_VALID_RETURN(pCtxInt);
     
    330378        {
    331379            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            }
    333393
    334394            pReqInt->AioCB.aio_sigevent.sigev_notify_kqueue = pCtxInt->iKQueue;
    335395            pReqInt->pCtxInt                                = pCtxInt;
     396            RTFILEAIOREQ_SET_STATE(pReqInt, SUBMITTED);
    336397
    337398            if (pReqInt->fFlush)
     
    346407            rcBSD = lio_listio(LIO_NOWAIT, (struct aiocb **)pahReqs, cReqsSubmit, NULL);
    347408            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            }
    349437
    350438            ASMAtomicAddS32(&pCtxInt->cRequests, cReqsSubmit);
    351439            cReqs   -= cReqsSubmit;
    352440            pahReqs += cReqsSubmit;
    353             *pcReqs += cReqsSubmit;
    354441        }
    355442
     
    368455                 rcBSD = aio_fsync(O_SYNC, &pReqInt->AioCB);
    369456                 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                 }
    371463
    372464                ASMAtomicIncS32(&pCtxInt->cRequests);
    373465                cReqs--;
    374466                pahReqs++;
    375                 *pcReqs++;
    376467            }
    377468        }
    378469    } while (cReqs);
    379470
    380     return VINF_SUCCESS;
     471    return rc;
    381472}
    382473
     
    447538        {
    448539            PRTFILEAIOREQINTERNAL pReqInt = (PRTFILEAIOREQINTERNAL)aKEvents[i].udata;
     540            AssertPtr(pReqInt);
     541            Assert(pReqInt->u32Magic == RTFILEAIOREQ_MAGIC);
    449542
    450543            /*
     
    468561                pReqInt->cbTransfered = cbTransfered;
    469562            }
     563            RTFILEAIOREQ_SET_STATE(pReqInt, COMPLETED);
    470564            pahReqs[cRequestsCompleted++] = (RTFILEAIOREQ)pReqInt;
    471565        }
Note: See TracChangeset for help on using the changeset viewer.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette