VirtualBox

Changeset 2307 in kBuild for trunk/src


Ignore:
Timestamp:
Mar 1, 2009 9:48:04 AM (16 years ago)
Author:
bird
Message:

kash: More file inheritance stuff.

Location:
trunk/src/kash
Files:
2 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/kash/shfile.c

    r2303 r2307  
    8484# define FDEV               0x40
    8585# define FTEXT              0x80
    86 #endif
     86
     87# define MY_ObjectBasicInformation 0
     88typedef LONG (NTAPI * PFN_NtQueryObject)(HANDLE, int, void *, size_t, size_t *);
     89
     90typedef struct MY_OBJECT_BASIC_INFORMATION
     91{
     92    ULONG           Attributes;
     93        ACCESS_MASK     GrantedAccess;
     94        ULONG           HandleCount;
     95        ULONG           PointerCount;
     96        ULONG           PagedPoolUsage;
     97        ULONG           NonPagedPoolUsage;
     98        ULONG           Reserved[3];
     99        ULONG           NameInformationLength;
     100        ULONG           TypeInformationLength;
     101        ULONG           SecurityDescriptorLength;
     102        LARGE_INTEGER   CreateTime;
     103} MY_OBJECT_BASIC_INFORMATION;
     104
     105#endif /* K_OS_WINDOWS */
    87106
    88107
     
    117136 * @param   pfdtab      The file descriptor table.
    118137 * @param   native      The native file handle.
    119  * @param   flags       The flags the it was created with.
     138 * @param   oflags      The flags the it was opened/created with.
     139 * @param   shflags     The shell file flags.
    120140 * @param   fdMin       The minimum file descriptor number, pass -1 if any is ok.
    121141 * @param   who         Who we're doing this for (for logging purposes).
    122142 */
    123 static int shfile_insert(shfdtab *pfdtab, intptr_t native, unsigned flags, int fdMin, const char *who)
     143static int shfile_insert(shfdtab *pfdtab, intptr_t native, unsigned oflags, unsigned shflags, int fdMin, const char *who)
    124144{
    125145    shmtxtmp tmp;
     
    164184            {
    165185                new_tab[i].fd = -1;
    166                 new_tab[i].flags = 0;
     186                new_tab[i].oflags = 0;
     187                new_tab[i].shflags = 0;
    167188                new_tab[i].native = -1;
    168189            }
     
    183204    {
    184205        pfdtab->tab[fd].fd = fd;
    185         pfdtab->tab[fd].flags = flags;
    186         pfdtab->tab[fd].cloexec = 0;
     206        pfdtab->tab[fd].oflags = oflags;
     207        pfdtab->tab[fd].shflags = shflags;
    187208        pfdtab->tab[fd].native = native;
    188209    }
    189210    else
    190         shfile_native_close(native, flags);
     211        shfile_native_close(native, oflags);
    191212
    192213    shmtx_leave(&pfdtab->mtx, &tmp);
     
    305326
    306327# if K_OS == K_OS_WINDOWS
     328
    307329/**
    308330 * Converts a DOS(/Windows) error code to errno,
     
    364386    return -1;
    365387}
     388
     389DWORD shfile_query_handle_access_mask(HANDLE h, PACCESS_MASK pMask)
     390{
     391    static PFN_NtQueryObject        s_pfnNtQueryObject = NULL;
     392    MY_OBJECT_BASIC_INFORMATION     BasicInfo;
     393    LONG                            Status;
     394
     395    if (!s_pfnNtQueryObject)
     396    {
     397        s_pfnNtQueryObject = (PFN_NtQueryObject)GetProcAddress(GetModuleHandle("NTDLL"), "NtQueryObject");
     398        if (!s_pfnNtQueryObject)
     399            return ERROR_NOT_SUPPORTED;
     400    }
     401
     402    Status = s_pfnNtQueryObject(h, MY_ObjectBasicInformation, &BasicInfo, sizeof(BasicInfo), NULL);
     403    if (Status >= 0)
     404    {
     405        *pMask = BasicInfo.GrantedAccess;
     406        return NO_ERROR;
     407    }
     408    if (Status != STATUS_INVALID_HANDLE)
     409        return ERROR_GEN_FAILURE;
     410    return ERROR_INVALID_HANDLE;
     411}
     412
    366413# endif /* K_OS == K_OS_WINDOWS */
    367414
     
    397444                {
    398445                    DWORD dwStdHandle;
    399                     unsigned flags;
     446                    unsigned fFlags;
    400447                } aStdHandles[3] =
    401448                {
     
    404451                    { STD_ERROR_HANDLE,   _O_WRONLY }
    405452                };
    406                 unsigned            i;
    407                 STARTUPINFO         Info;
     453                int             i;
     454                STARTUPINFO     Info;
     455                ACCESS_MASK     Mask;
     456                DWORD           dwErr;
    408457
    409458                rc = 0;
     
    415464                    memset(&Info, 0, sizeof(Info));
    416465                }
    417                 if (    Info.cbReserved2 <= sizeof(int)
     466
     467                if (    Info.cbReserved2 > sizeof(int)
    418468                    &&  (uintptr_t)Info.lpReserved2 >= 0x1000
    419                     &&  *(int *)Info.lpReserved2 * (sizeof(int) + sizeof(intptr_t)) + 4 <= Info.cbReserved2
    420                     &&  *(int *)Info.lpReserved2 <= 2048
    421                     &&  *(int *)Info.lpReserved2 >= 1)
     469                    &&  (i = *(int *)Info.lpReserved2) >= 1
     470                    &&  i <= 2048
     471                    &&  (   Info.cbReserved2 == i * 5 + 4
     472                         //|| Info.cbReserved2 == i * 5 + 1 - check the cygwin sources.
     473                         || Info.cbReserved2 == i * 9 + 4))
    422474                {
    423                     unsigned    c       = *(int *)Info.lpReserved2;
    424                     char       *aosfile = (char *)Info.lpReserved2 + sizeof(int);
    425                     intptr_t   *aosfhnd = (intptr_t *)(aosfile + c);
    426 
    427                     /** @todo process */
    428 
     475                    uint8_t    *paf    = (uint8_t *)Info.lpReserved2 + sizeof(int);
     476                    int         dwPerH = 1 + (Info.cbReserved2 == i * 9 + 4);
     477                    DWORD      *ph     = (DWORD *)(paf + i) + dwPerH * i;
     478                    HANDLE      aStdHandles2[3];
     479                    int         j;
     480
     481                    //if (Info.cbReserved2 == i * 5 + 1) - check the cygwin sources.
     482                    //    i--;
     483
     484                    for (j = 0; j < 3; j++)
     485                        aStdHandles2[j] = GetStdHandle(aStdHandles[j].dwStdHandle);
     486
     487                    while (i-- > 0)
     488                    {
     489                        ph -= dwPerH;
     490
     491                        if (    (paf[i] & (FOPEN | FNOINHERIT)) == FOPEN
     492                            &&  *ph != (uint32_t)INVALID_HANDLE_VALUE)
     493                        {
     494                            HANDLE  h = (HANDLE)(intptr_t)*ph;
     495                            int     fd2;
     496                            int     fFlags;
     497                            int     fFlags2;
     498
     499                            if (    h == aStdHandles2[j = 0]
     500                                ||  h == aStdHandles2[j = 1]
     501                                ||  h == aStdHandles2[j = 2])
     502                                fFlags = aStdHandles[j].fFlags;
     503                            else
     504                            {
     505                                dwErr = shfile_query_handle_access_mask(h, &Mask);
     506                                if (dwErr == ERROR_INVALID_HANDLE)
     507                                    continue;
     508                                else if (dwErr == NO_ERROR)
     509                                {
     510                                    fFlags = 0;
     511                                    if (    (Mask & (GENERIC_READ | FILE_READ_DATA))
     512                                        &&  (Mask & (GENERIC_WRITE | FILE_WRITE_DATA | FILE_APPEND_DATA)))
     513                                        fFlags |= O_RDWR;
     514                                    else if (Mask & (GENERIC_READ | FILE_READ_DATA))
     515                                        fFlags |= O_RDONLY;
     516                                    else if (Mask & (GENERIC_WRITE | FILE_WRITE_DATA | FILE_APPEND_DATA))
     517                                        fFlags |= O_WRONLY;
     518                                    else
     519                                        fFlags |= O_RDWR;
     520                                    if ((Mask & (FILE_WRITE_DATA | FILE_APPEND_DATA)) == FILE_APPEND_DATA)
     521                                        fFlags |= O_APPEND;
     522                                }
     523                                else
     524                                    fFlags = O_RDWR;
     525                            }
     526
     527                            if (paf[i] & FPIPE)
     528                                fFlags2 = SHFILE_FLAGS_PIPE;
     529                            else if (paf[i] & FDEV)
     530                                fFlags2 = SHFILE_FLAGS_TTY;
     531                            else
     532                                fFlags2 = 0;
     533
     534                            fd2 = shfile_insert(pfdtab, (intptr_t)h, fFlags, fFlags2, i, "shtab_init");
     535                            assert(fd2 == i); (void)fd2;
     536                            if (fd2 != i)
     537                                rc = -1;
     538                        }
     539                    }
    429540                }
    430541
    431542                /* Check the three standard handles. */
    432543                for (i = 0; i < 3; i++)
    433                 {
    434                     HANDLE hFile = GetStdHandle(aStdHandles[i].dwStdHandle);
    435                     if (    hFile != INVALID_HANDLE_VALUE
    436                         &&  (   (unsigned)i >= pfdtab->size
    437                              || pfdtab->tab[i].fd == -1))
     544                    if (    (unsigned)i >= pfdtab->size
     545                        ||  pfdtab->tab[i].fd == -1)
    438546                    {
    439                         int fd2 = shfile_insert(pfdtab, (intptr_t)hFile, aStdHandles[i].flags, i, "shtab_init");
    440                         assert(fd2 == i); (void)fd2;
    441                         if (fd2 != i)
    442                             rc = -1;
     547                        HANDLE hFile = GetStdHandle(aStdHandles[i].dwStdHandle);
     548                        if (hFile != INVALID_HANDLE_VALUE)
     549                        {
     550                            int fd2 = shfile_insert(pfdtab, (intptr_t)hFile, aStdHandles[i].fFlags, 0, i, "shtab_init");
     551                            assert(fd2 == i); (void)fd2;
     552                            if (fd2 != i)
     553                                rc = -1;
     554                        }
    443555                    }
    444                 }
    445556# else
    446557# endif
     
    486597            HANDLE hFile = (HANDLE)pfdtab->tab[i].native;
    487598            if (set)
    488                 TRACE2((NULL, "  #%d: native=%#x flags=%#x cloexec=%d\n",
    489                         i, pfdtab->tab[i].flags, hFile, pfdtab->tab[i].cloexec));
     599                TRACE2((NULL, "  #%d: native=%#x oflags=%#x shflags=%#x\n",
     600                        i, pfdtab->tab[i].oflags, pfdtab->tab[i].shflags, hFile));
    490601            if (!SetHandleInformation(hFile, HANDLE_FLAG_INHERIT, fFlag))
    491602            {
     
    538649           ? pfdtab->size
    539650           : (0x10000-4) / (1 + sizeof(HANDLE));
     651    while (count > 3 && pfdtab->tab[count].fd == -1)
     652        count--;
    540653
    541654    if (prepare)
     
    552665        {
    553666            if (    pfdtab->tab[i].fd == i
    554                 &&  !pfdtab->tab[i].cloexec)
     667                &&  !(pfdtab->tab[i].shflags & SHFILE_FLAGS_CLOSE_ON_EXEC))
    555668            {
    556669                HANDLE hFile = (HANDLE)pfdtab->tab[i].native;
    557                 TRACE2((NULL, "  #%d: native=%#x flags=%#x\n",
    558                         i, pfdtab->tab[i].flags, hFile));
     670                TRACE2((NULL, "  #%d: native=%#x oflags=%#x shflags=%#x\n",
     671                        i, pfdtab->tab[i].oflags, pfdtab->tab[i].shflags, hFile));
    559672
    560673                if (!SetHandleInformation(hFile, HANDLE_FLAG_INHERIT, HANDLE_FLAG_INHERIT))
     
    564677                }
    565678                paf[i] = FOPEN;
    566                 if (pfdtab->tab[i].flags & _O_APPEND)
    567                     paf[i] = FAPPEND;
    568                 if (pfdtab->tab[i].flags & _O_TEXT)
    569                     paf[i] = FTEXT;
     679                if (pfdtab->tab[i].oflags & _O_APPEND)
     680                    paf[i] |= FAPPEND;
     681                if (pfdtab->tab[i].oflags & _O_TEXT)
     682                    paf[i] |= FTEXT;
     683                switch (pfdtab->tab[i].shflags & SHFILE_FLAGS_TYPE_MASK)
     684                {
     685                    case SHFILE_FLAGS_TTY:  paf[i] |= FDEV; break;
     686                    case SHFILE_FLAGS_PIPE: paf[i] |= FPIPE; break;
     687                }
    570688                pah[i] = hFile;
    571689            }
     
    597715        {
    598716            if (    pfdtab->tab[i].fd == i
    599                 &&  !pfdtab->tab[i].cloexec)
     717                &&  !(pfdtab->tab[i].shflags & SHFILE_FLAGS_CLOSE_ON_EXEC))
    600718            {
    601719                HANDLE hFile = (HANDLE)pfdtab->tab[i].native;
     
    689807                            NULL /* hTemplateFile */);
    690808        if (hFile != INVALID_HANDLE_VALUE)
    691             fd = shfile_insert(pfdtab, (intptr_t)hFile, flags, -1, "shfile_open");
     809            fd = shfile_insert(pfdtab, (intptr_t)hFile, flags, 0, -1, "shfile_open");
    692810        else
    693811            fd = shfile_dos2errno(GetLastError());
     
    706824            errno = s;
    707825            if (native != -1)
    708                 fd = shfile_insert(pfdtab, native, flags, -1, "shfile_open");
     826                fd = shfile_insert(pfdtab, native, flags, 0, -1, "shfile_open");
    709827            else
    710828                fd = -1;
     
    739857    if (CreatePipe(&hRead, &hWrite, &SecurityAttributes, 4096))
    740858    {
    741         fds[0] = shfile_insert(pfdtab, (intptr_t)hRead, O_RDONLY, -1, "shfile_pipe");
     859        fds[0] = shfile_insert(pfdtab, (intptr_t)hRead, O_RDONLY, SHFILE_FLAGS_PIPE, -1, "shfile_pipe");
    742860        if (fds[0] != -1)
    743861        {
    744             fds[1] = shfile_insert(pfdtab, (intptr_t)hWrite, O_WRONLY, -1, "shfile_pipe");
     862            fds[1] = shfile_insert(pfdtab, (intptr_t)hWrite, O_WRONLY, SHFILE_FLAGS_PIPE, -1, "shfile_pipe");
    745863            if (fds[1] != -1)
    746864                rc = 0;
     
    753871    if (!pipe(native_fds))
    754872    {
    755         fds[0] = shfile_insert(pfdtab, native_fds[0], O_RDONLY, -1, "shfile_pipe");
     873        fds[0] = shfile_insert(pfdtab, native_fds[0], O_RDONLY, SHFILE_FLAGS_PIPE, -1, "shfile_pipe");
    756874        if (fds[0] != -1)
    757875        {
    758             fds[1] = shfile_insert(pfdtab, native_fds[1], O_WRONLY, -1, "shfile_pipe");
     876            fds[1] = shfile_insert(pfdtab, native_fds[1], O_WRONLY, SHFILE_FLAGS_PIPE, -1, "shfile_pipe");
    759877            if (fds[1] != -1)
    760878                rc = 0;
     
    770888                rc = fds[0];
    771889                pfdtab->tab[rc].fd = -1;
    772                 pfdtab->tab[rc].flags = 0;
     890                pfdtab->tab[rc].oflags = 0;
     891                pfdtab->tab[rc].shflags = 0;
    773892                pfdtab->tab[rc].native = -1;
    774893                shmtx_leave(&pfdtab->mtx, &tmp);
     
    822941    if (file)
    823942    {
    824         shfile_native_close(file->native, file->flags);
     943        shfile_native_close(file->native, file->oflags);
    825944
    826945        file->fd = -1;
    827         file->flags = 0;
     946        file->oflags = 0;
     947        file->shflags = 0;
    828948        file->native = -1;
    829949
     
    9511071        {
    9521072            case F_GETFL:
    953                 rc = file->flags;
     1073                rc = file->oflags;
    9541074                break;
    9551075
     
    9661086                mask |= O_SYNC;
    9671087# endif
    968                 if ((file->flags & mask) == (arg & mask))
     1088                if ((file->oflags & mask) == (arg & mask))
    9691089                    rc = 0;
    9701090                else
     
    9941114                                    FALSE /* bInheritHandle */,
    9951115                                    DUPLICATE_SAME_ACCESS))
    996                     rc = shfile_insert(pfdtab, (intptr_t)hNew, file->flags, arg, "shfile_fcntl");
     1116                    rc = shfile_insert(pfdtab, (intptr_t)hNew, file->oflags, file->shflags, arg, "shfile_fcntl");
    9971117                else
    9981118                    rc = shfile_dos2errno(GetLastError());
     
    10001120                int nativeNew = fcntl(file->native F_DUPFD, SHFILE_UNIX_MIN_FD);
    10011121                if (nativeNew != -1)
    1002                     rc = shfile_insert(pfdtab, nativeNew, file->flags, arg, "shfile_fcntl");
     1122                    rc = shfile_insert(pfdtab, nativeNew, file->oflags, file->shflags, arg, "shfile_fcntl");
    10031123# endif
    10041124                break;
     
    12041324    if (file)
    12051325    {
    1206         file->cloexec = !!(closeit);
     1326        if (closeit)
     1327            file->shflags |= SHFILE_FLAGS_CLOSE_ON_EXEC;
     1328        else
     1329            file->shflags &= ~SHFILE_FLAGS_CLOSE_ON_EXEC;
    12071330        shfile_put(pfdtab, file, &tmp);
    12081331    }
  • trunk/src/kash/shfile.h

    r2303 r2307  
    102102{
    103103    int                 fd;             /**< The shell file descriptor number. */
    104     int                 flags;          /**< Open flags. */
    105     unsigned            cloexec : 1;    /**< Close on exec flag. */
     104    unsigned            oflags;         /**< Open flags. */
     105    unsigned            shflags;        /**< The shell file descriptor flags. */
    106106    intptr_t            native;         /**< The native file descriptor number. */
    107107} shfile;
     108
     109/** @name shfile::shflags values.
     110 * @{
     111 */
     112#define SHFILE_FLAGS_CLOSE_ON_EXEC      0x0001
     113#define SHFILE_FLAGS_TYPE_MASK          0x00f0
     114#define SHFILE_FLAGS_FILE               0x0000
     115#define SHFILE_FLAGS_PIPE               0x0010
     116#define SHFILE_FLAGS_DIR                0x0020
     117#define SHFILE_FLAGS_TTY                0x0030
     118/** @} */
    108119
    109120/**
Note: See TracChangeset for help on using the changeset viewer.

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