df.c

来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 288 行

C
288
字号
/****************************************************************************
*
*                            Open Watcom Project
*
*    Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
*
*  ========================================================================
*
*    This file contains Original Code and/or Modifications of Original
*    Code as defined in and that are subject to the Sybase Open Watcom
*    Public License version 1.0 (the 'License'). You may not use this file
*    except in compliance with the License. BY USING THIS FILE YOU AGREE TO
*    ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
*    provided with the Original Code and Modifications, and is also
*    available at www.sybase.com/developer/opensource.
*
*    The Original Code and all software distributed under the License are
*    distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
*    EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
*    ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
*    MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
*    NON-INFRINGEMENT. Please see the License for the specific language
*    governing rights and limitations under the License.
*
*  ========================================================================
*
* Description:  POSIX df utility
*               Displays disk free space
*
****************************************************************************/


#if defined( __OS_os2386__ ) || defined( __OS_dosos2__ )
#include <os2.h>
#elif defined( __OS_nt386__ ) || defined( __OS_ntaxp__ )
#include <windows.h>
#endif
#include <stdio.h>
#include <stdlib.h>
#include <dos.h>
#include <string.h>
#include <ctype.h>
#include "misc.h"
#include "getopt.h"
#include "argvenv.h"

typedef enum {
    DRIVE_NONE,
    DRIVE_IS_REMOVABLE,
    DRIVE_IS_FIXED,
    DRIVE_IS_REMOTE
} drive_type;

static char     includeNetwork;

char            *OptEnvVar = "df";

static const char *usageMsg[] = {
    "Usage: df [-?n] [@env] [drive list]",
    "\tenv            : environment variable to expand",
    "\tdrive list     : specify which drives do show",
    "\tOptions: -?    : print this list",
    "\t\t -n    : include network drives",
    NULL
};

#if defined( __OS_dos__ ) || defined( __OS_dosos2__ ) || defined( __OS_pharlap__ )
extern short CheckRemovable( char );
#pragma aux CheckRemovable = \
        "mov    ax,04408h" \
        "int    021h" \
        "cmp    ax,0fh" \
        "jne    ok" \
        "mov    ax,0" \
        "jmp    done" \
        "ok:    inc ax" \
        "done:" \
        parm [bl] value[ax];

extern short CheckRemote( char );
#pragma aux CheckRemote = \
        "mov    ax,04409h" \
        "int    021h" \
        "mov    ax,-1" \
        "jc     ibrx" \
        "inc    ax" \
        "test   dh,10h" \
        "jz     ibrx" \
        "inc    ax" \
        "ibrx:" \
        parm [bl] value[ax] modify[dx];

/*
 * doGetDriveType - get the type of drive A-Z
 */
#ifdef __OS_dos__
static drive_type doGetDriveType( int drv )
#else
static drive_type dosDoGetDriveType( int drv )
#endif
{
    int         dl;
    drive_type  rc;

    dl = toupper( drv ) - 'A' + 1;
    rc = CheckRemovable( dl );
    if( rc == DRIVE_IS_FIXED ) {
        if( CheckRemote( dl ) ) {
            rc = DRIVE_IS_REMOTE;
        }
    }
    return( rc );

} /* doGetDriveType */
#endif

#if defined( __OS_os2386__ ) || defined( __OS_dosos2__ )

#ifdef __OS_os2386__
#define STUPID_UINT     unsigned long
#else
#define STUPID_UINT     unsigned short
#endif
/*
 * doGetDriveType - get the type of drive A-Z
 */
static drive_type doGetDriveType( int drv )
{
    STUPID_UINT disk;
    ULONG       map;
    int         i;
    char        fname[3];
    HFILE       hf;
    STUPID_UINT act;
    STUPID_UINT type;
    int         rc;

#ifdef __OS_dosos2__
    if( _osmode == 0 ) {
        return( dosDoGetDriveType( drv ) );
    } else {
#endif
        drv = toupper( drv );
        DosQCurDisk( &disk, &map );
        for( i='A';i<='Z';i++ ) {
            if( drv == i ) {
                if( map & 1 ) {
                    fname[0] = tolower( i );
                    fname[1] = ':';
                    fname[2] = 0;
                    rc = DosOpen( fname, &hf, &act, 0, 0,
                                0x0001,
                                0x0040 | 0x8000, 0L );
                    if( rc == 0 ) {
                        DosQHandType( hf, &type, &act );
                        DosClose( hf );
                        if( type & 0x8000 ) {
                            return( DRIVE_IS_REMOTE );
                        }
                    } else {
                        // KLUDGE - this only seems to fail for network drives?!
                        return( DRIVE_IS_REMOTE );
                    }
                    return( DRIVE_IS_FIXED );
                } else {
                    return( DRIVE_NONE );
                }
            }
            map >>= 1;
        }
        return( DRIVE_NONE );
#ifdef __OS_dosos2__
    }
#endif

} /* doGetDriveType */

unsigned _dos_getdiskfree( unsigned dnum, struct diskfree_t *df )
{
    FSALLOCATE  fs;
    unsigned    rc;

    rc = DosQFSInfo( dnum, 1, (PVOID) &fs, sizeof( fs ) );
    if( rc ) {
        return( rc );
    }
    df->sectors_per_cluster = fs.cSectorUnit;
    df->total_clusters = fs.cUnit;
    df->avail_clusters = fs.cUnitAvail;
    df->bytes_per_sector = fs.cbSector;
    return( 0 );
}

#elif defined( __OS_nt386__ ) || defined( __OS_ntaxp__ )

/*
 * doGetDriveType - get the type of drive A-Z
 */
static drive_type doGetDriveType( int drv )
{
    char        path[4];
    DWORD       type;

    path[0] = drv;
    path[1] = ':';
    path[2] = '\\';
    path[3] = 0;
    type = GetDriveType( path );
    if( type == 1 ) {
        return( DRIVE_NONE );
    }
    if( type == DRIVE_REMOVABLE ) {
        return( DRIVE_IS_REMOVABLE );
    }
    if( type == DRIVE_REMOTE ) {
        return( DRIVE_IS_REMOTE );
    }
    return( DRIVE_IS_FIXED );

} /* doGetDriveType */
#endif

/*
 * doDF - do disk info for a specific drive
 */
static void doDF( int drive )
{
    struct diskfree_t   df;
    unsigned long long  cl_bytes;
    long                total;
    long                avail;
    long                used;
    drive_type          dt;

    dt = doGetDriveType( drive );
    if( dt == DRIVE_NONE ) {
        return;
    }
    if( dt == DRIVE_IS_REMOTE && !includeNetwork ) {
        return;
    }
    if( _dos_getdiskfree( toupper( drive ) - 'A' +1, &df ) == 0 ) {
        cl_bytes = (long)df.sectors_per_cluster * (long)df.bytes_per_sector;
        total = (long)df.total_clusters * cl_bytes / 1024L;
        avail = (long)df.avail_clusters * cl_bytes / 1024L;
        used = total-avail;
        printf( " %c   %10ld  %10ld  %10ld    %2d%%\n",
            toupper( drive ), total, used, avail, (100L*used)/total );
    }

} /* doDF */


void main( int argc, char *argv[] )
{
    int         ch;
    int         i;

    argv = ExpandEnv( &argc, argv );

    while( 1 ) {
        ch = GetOpt( &argc, argv, "n", usageMsg );
        if( ch == -1 ) {
            break;
        }
        switch( ch ) {
        case 'n':
            includeNetwork = 1;
            break;
        }
    }

    printf( "drive  KBytes        used       avail   capacity\n" );
    if( argc > 1 ) {
        includeNetwork = 1;
        for( i = 1; i < argc; i++ ) {
            doDF( argv[i][0] );
        }
    } else {
        for( i = 'C'; i <= 'Z'; i++ ) {
            doDF( i );
        }
    }

    exit( 0 );

} /* main */

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?