590 lines
19 KiB
C
Raw Permalink Normal View History

2001-01-01 00:00:00 +01:00
// du - simple disk usage program
// If UNICODE/_UNICODE is turned on, we need to link with
// wsetargv.lib (not setargv.lib) and with UMENTRY=wmain
#define UNICODE
#define _UNICODE
#include <stdio.h>
#include <tchar.h>
#include <wchar.h>
#include <string.h>
#include <process.h>
#include <ctype.h>
#include <malloc.h>
#include <stdlib.h>
#include <locale.h>
#include <windows.h>
typedef struct USESTAT USESTAT;
typedef struct EXTSTAT EXTSTAT;
typedef USESTAT *PUSESTAT;
struct USESTAT {
DWORDLONG cchUsed; // bytes used in all files
DWORDLONG cchAlloc; // bytes allocated in all files
DWORDLONG cchCompressed; // compressed bytes in all files
DWORDLONG cchDeleted; // bytes in deleted files
DWORDLONG cFile; // number of files
};
struct EXTSTAT {
EXTSTAT *Next;
TCHAR *Extension;
USESTAT Stat;
};
EXTSTAT *ExtensionList = NULL;
int ExtensionCount = 0;
#define CLEARUSE(use) \
{ (use).cchUsed = (DWORDLONG)0; \
(use).cchAlloc = (DWORDLONG)0; \
(use).cchDeleted = (DWORDLONG)0; \
(use).cchCompressed = (DWORDLONG)0; \
(use).cFile = (DWORDLONG)0; \
}
#define ADDUSE(sum,add) \
{ (sum).cchUsed += (add).cchUsed; \
(sum).cchAlloc += (add).cchAlloc; \
(sum).cchDeleted += (add).cchDeleted; \
(sum).cchCompressed += (add).cchCompressed; \
(sum).cFile += (add).cFile; \
}
#define DWORD_SHIFT (sizeof(DWORD) * 8)
#define SHIFT(c,v) {c--; v++;}
DWORD gdwOutputMode;
HANDLE ghStdout;
int cDisp; // number of summary lines displayed
BOOL fExtensionStat = FALSE; // TRUE gather statistics by extension
BOOL fNodeSummary = FALSE; // TRUE => only display top-level
BOOL fShowDeleted = FALSE; // TRUE => show deleted files information
BOOL fThousandSeparator = TRUE; // TRUE => use thousand separator in output
BOOL fShowCompressed = FALSE; // TRUE => show compressed file info
BOOL fSubtreeTotal = FALSE; // TRUE => show info in subtree total form (add from bottom up)
BOOL fUnc = FALSE; // Set if we're checking a UNC path.
TCHAR *pszDeleted = TEXT("deleted\\*.*");
TCHAR *pszWild = TEXT("*.*");
long bytesPerAlloc;
int bValidDrive;
DWORDLONG totFree;
DWORDLONG totDisk;
TCHAR buf[MAX_PATH];
TCHAR root[] = TEXT("?:\\");
USESTAT DoDu (TCHAR *dir);
void TotPrint (PUSESTAT puse, TCHAR *p);
TCHAR ThousandSeparator[8];
TCHAR *
FormatFileSize(
DWORDLONG FileSize,
TCHAR *FormattedSize,
ULONG Width
)
{
TCHAR Buffer[ 100 ];
TCHAR *s, *s1;
ULONG DigitIndex, Digit;
ULONG nThousandSeparator;
DWORDLONG Size;
nThousandSeparator = _tcslen(ThousandSeparator);
s = &Buffer[ 99 ];
*s = TEXT('\0');
DigitIndex = 0;
Size = FileSize;
while (Size != 0) {
Digit = (ULONG)(Size % 10);
Size = Size / 10;
*--s = (TCHAR)(TEXT('0') + Digit);
if ((++DigitIndex % 3) == 0 && fThousandSeparator) {
// If non-null Thousand separator, insert it.
if (nThousandSeparator) {
s -= nThousandSeparator;
_tcsncpy(s, ThousandSeparator, nThousandSeparator);
}
}
}
if (DigitIndex == 0) {
*--s = TEXT('0');
}
else
if (fThousandSeparator && !_tcsncmp(s, ThousandSeparator, nThousandSeparator)) {
s += nThousandSeparator;
}
Size = _tcslen( s );
if (Width != 0 && Size < Width) {
s1 = FormattedSize;
while (Width > Size) {
Width -= 1;
*s1++ = TEXT(' ');
}
_tcscpy( s1, s );
} else {
_tcscpy( FormattedSize, s );
}
return FormattedSize;
}
#ifdef UNICODE
int __cdecl wmain(int c, wchar_t **v, wchar_t **envp)
#else
int __cdecl main(int c, char *v[])
#endif
{
int tenth, pct;
int bValidBuf;
DWORDLONG tmpTot, tmpFree;
DWORD cSecsPerClus, cBytesPerSec, cFreeClus, cTotalClus;
USESTAT useTot, useTmp;
TCHAR Buffer[MAX_PATH];
TCHAR *p;
UINT Codepage;
char achCodepage[6] = ".OCP";
ghStdout = GetStdHandle(STD_OUTPUT_HANDLE);
GetConsoleMode(ghStdout, &gdwOutputMode);
gdwOutputMode &= ~ENABLE_PROCESSED_OUTPUT;
/*
* This is mainly here as a good example of how to set a character-mode
* application's codepage.
* This affects C-runtime routines such as mbtowc(), mbstowcs(), wctomb(),
* wcstombs(), mblen(), _mbstrlen(), isprint(), isalpha() etc.
* To make sure these C-runtimes come from msvcrt.dll, use TARGETLIBS in
* the sources file, together with TARGETTYPE=PROGRAM (and not UMAPPL?)
*/
if (Codepage = GetConsoleOutputCP()) {
sprintf(achCodepage, ".%3.4d", Codepage);
}
setlocale(LC_ALL, achCodepage);
SHIFT (c, v);
if (GetLocaleInfo(GetUserDefaultLCID(),
LOCALE_STHOUSAND,
Buffer,
sizeof(ThousandSeparator)/sizeof(TCHAR))) {
#ifdef UNICODE
_tcscpy(ThousandSeparator, Buffer);
#else
CharToOemA(Buffer, ThousandSeparator);
#endif
}
else {
_tcscpy(ThousandSeparator, TEXT(","));
}
while (c && (**v == TEXT('/') || **v == TEXT('-')))
{
if (!_tcscmp (*v + 1, TEXT("e"))) {
fExtensionStat = TRUE;
} else
if (!_tcscmp (*v + 1, TEXT("s")))
fNodeSummary = TRUE;
else
if (!_tcscmp (*v + 1, TEXT("d")))
fShowDeleted = TRUE;
else
if (!_tcscmp (*v + 1, TEXT("p")))
fThousandSeparator = FALSE;
else
if (!_tcscmp (*v + 1, TEXT("c")))
fShowCompressed = TRUE;
else
if (!_tcscmp (*v + 1, TEXT("t")))
fSubtreeTotal = TRUE;
else
{
_fputts( TEXT("Usage: DU [/e] [/d] [/p] [/s] [/c] [/t] [dirs]\n")
TEXT("where:\n")
TEXT(" /e - displays information by extension.\n")
TEXT(" /d - displays informations about [deleted] subdirectories.\n")
TEXT(" /p - displays numbers plainly, without thousand separators.\n")
TEXT(" /s - displays summary information only.\n")
TEXT(" /c - displays compressed file information.\n")
TEXT(" /t - displays information in subtree total form.\n"),
stderr);
exit (1);
}
SHIFT (c, v);
}
if (c == 0)
{
GetCurrentDirectory( MAX_PATH, (LPTSTR)buf );
root[0] = buf[0];
if( bValidDrive = GetDiskFreeSpace( root,
&cSecsPerClus,
&cBytesPerSec,
&cFreeClus,
&cTotalClus ) == TRUE )
{
bytesPerAlloc = cBytesPerSec * cSecsPerClus;
totFree = (DWORDLONG)bytesPerAlloc * cFreeClus;
totDisk = (DWORDLONG)bytesPerAlloc * cTotalClus;
}
useTot = DoDu (buf);
if (fNodeSummary)
TotPrint (&useTot, buf);
}
else
{
CLEARUSE (useTot);
while (c)
{
LPTSTR FilePart;
bValidBuf = GetFullPathName( *v, MAX_PATH, buf, &FilePart);
if ( bValidBuf )
{
if ( buf[0] == TEXT('\\') ) {
fUnc = TRUE;
bValidDrive = TRUE;
bytesPerAlloc = 1;
} else {
root[0] = buf[0];
if( bValidDrive = GetDiskFreeSpace( root,
&cSecsPerClus,
&cBytesPerSec,
&cFreeClus,
&cTotalClus ) == TRUE)
{
bytesPerAlloc = cBytesPerSec * cSecsPerClus;
totFree = (DWORDLONG)bytesPerAlloc * cFreeClus;
totDisk = (DWORDLONG)bytesPerAlloc * cTotalClus;
} else
_tprintf (TEXT("Invalid drive or directory %s\n"), *v );
}
if( bValidDrive && (GetFileAttributes( buf ) & FILE_ATTRIBUTE_DIRECTORY ) != 0 )
{
useTmp = DoDu (buf);
if (fNodeSummary)
TotPrint (&useTmp, buf);
ADDUSE (useTot, useTmp);
}
}
else
_tprintf (TEXT("Invalid drive or directory %s\n"), *v );
SHIFT (c, v);
}
}
if (cDisp != 0)
{
if (cDisp > 1)
TotPrint (&useTot, TEXT("Total"));
/* quick full-disk test */
if ( !fUnc ) {
if (totFree == 0)
_putts (TEXT("Disk is full"));
else
{
tmpTot = (totDisk + 1023) / 1024;
tmpFree = (totFree + 1023) / 1024;
pct = (DWORD)(1000 * (tmpTot - tmpFree) / tmpTot);
tenth = pct % 10;
pct /= 10;
// Disable processing so Middle Dot won't beep
// Middle Dot 0x2022 aliases to ^G when using Raster Fonts
SetConsoleMode(ghStdout, gdwOutputMode);
_tprintf(TEXT("%s/"), FormatFileSize( totDisk-totFree, Buffer, 0 ));
_tprintf(TEXT("%s "), FormatFileSize( totDisk, Buffer, 0 ));
// Re-enable processing so newline works
SetConsoleMode(ghStdout, gdwOutputMode | ENABLE_PROCESSED_OUTPUT);
_tprintf (TEXT("%d.%d%% of disk in use\n"), pct, tenth);
}
}
}
if (fExtensionStat) {
int i;
_tprintf( TEXT("\n") );
for (i = 0; i < ExtensionCount; i++) {
TotPrint( &ExtensionList[i].Stat, ExtensionList[i].Extension );
}
}
return( 0 );
}
int __cdecl ExtSearchCompare( const void *Key, const void *Element)
{
return _tcsicmp( (TCHAR *)Key, ((EXTSTAT *) Element)->Extension );
}
int __cdecl ExtSortCompare( const void *Element1, const void *Element2)
{
return _tcsicmp( ((EXTSTAT *) Element1)->Extension, ((EXTSTAT *) Element2)->Extension );
}
#define MYMAKEDWORDLONG(h,l) (((DWORDLONG)(h) << DWORD_SHIFT) + (DWORDLONG)(l))
#define FILESIZE(wfd) MYMAKEDWORDLONG((wfd).nFileSizeHigh, (wfd).nFileSizeLow)
#define ROUNDUP(m,n) ((((m) + (n) - 1) / (n)) * (n))
// Count the number of useable characters remaining in a null terminated string
// s of buffer length cch beyond and including a point specified by p
#define REMAINING_STRING(s, cch, p) (cch - (p - s) - 1)
USESTAT DoDu (TCHAR *dir)
{
WIN32_FIND_DATA wfd;
HANDLE hFind;
USESTAT use, DirUse;
TCHAR pszSearchName[MAX_PATH];
TCHAR *pszFilePart;
DWORDLONG compressedSize;
DWORD compHi, compLo;
SIZE_T remaining;
CLEARUSE(use);
// Make a copy of the incoming directory name and append a trailing
// slash if necessary. pszFilePart will point to the char just after
// the slash, making it easy to build fully qualified filenames.
//
// Slap a null at the end of the string since strncpy doesn't.
_tcsncpy(pszSearchName, dir, sizeof(pszSearchName)/sizeof(TCHAR) - 1);
pszSearchName[sizeof(pszSearchName)/sizeof(TCHAR) - 1] = TEXT('\0');
pszFilePart = pszSearchName + _tcslen(pszSearchName);
remaining = REMAINING_STRING(pszSearchName,
sizeof(pszSearchName)/sizeof(TCHAR),
pszFilePart);
if (pszFilePart > pszSearchName)
{
if (pszFilePart[-1] != TEXT('\\') && pszFilePart[-1] != TEXT('/'))
{
// Give up if we don't have enough string left
if (!remaining) {
return (use);
}
*pszFilePart++ = TEXT('\\');
remaining -= 1;
}
}
if (fShowDeleted &&
remaining >= _tcslen(pszDeleted)) {
// First count the size of all the files in the current deleted tree
_tcscpy(pszFilePart, pszDeleted);
hFind = FindFirstFile(pszSearchName, &wfd);
if (hFind != INVALID_HANDLE_VALUE)
{
do
{
if (!(wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY))
{
use.cchDeleted += ROUNDUP( FILESIZE( wfd ), bytesPerAlloc );
}
} while (FindNextFile(hFind, &wfd));
FindClose(hFind);
}
}
// Give up if we can't put the wild chars at the end
if (remaining < _tcslen(pszWild)) {
return(use);
}
// Then count the size of all the file in the current tree.
_tcscpy(pszFilePart, pszWild);
hFind = FindFirstFile(pszSearchName, &wfd);
if (hFind != INVALID_HANDLE_VALUE)
{
do
{
if (!(wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY))
{
use.cchUsed += FILESIZE( wfd );
use.cchAlloc += ROUNDUP( FILESIZE( wfd ), bytesPerAlloc );
use.cFile++;
compressedSize = FILESIZE(wfd);
if (fShowCompressed && (wfd.dwFileAttributes & FILE_ATTRIBUTE_COMPRESSED))
{
_tcscpy(pszFilePart, wfd.cFileName);
compLo = GetCompressedFileSize(pszSearchName, &compHi);
if (compLo != (DWORD)-1 || GetLastError() == 0) {
compressedSize = MYMAKEDWORDLONG(compHi, compLo);
}
}
use.cchCompressed += compressedSize;
//
// Accrue statistics by extension
//
if (fExtensionStat) {
TCHAR Ext[_MAX_EXT];
EXTSTAT *ExtensionStat;
_tsplitpath( wfd.cFileName, NULL, NULL, NULL, Ext );
while (TRUE) {
//
// Find extension in list
//
ExtensionStat =
(EXTSTAT *) bsearch( Ext, ExtensionList,
ExtensionCount, sizeof( EXTSTAT ),
ExtSearchCompare );
if (ExtensionStat != NULL) {
break;
}
//
// Extension not found, go add one and resort
//
ExtensionCount++;
{
void *pv = realloc( ExtensionList, sizeof( EXTSTAT ) * ExtensionCount);
if (pv) {
ExtensionList = (EXTSTAT *)pv;
} else {
_putts (TEXT("Out of memory"));
}
}
ExtensionList[ExtensionCount - 1].Extension = _tcsdup( Ext );
CLEARUSE( ExtensionList[ExtensionCount - 1].Stat );
qsort( ExtensionList, ExtensionCount, sizeof( EXTSTAT ), ExtSortCompare );
}
ExtensionStat->Stat.cchUsed += FILESIZE( wfd );
ExtensionStat->Stat.cchAlloc += ROUNDUP( FILESIZE( wfd ), bytesPerAlloc );
ExtensionStat->Stat.cchCompressed += compressedSize;
ExtensionStat->Stat.cFile++;
}
}
} while (FindNextFile(hFind, &wfd));
FindClose(hFind);
}
if (!fNodeSummary && !fSubtreeTotal)
TotPrint (&use, dir);
// Now, do all the subdirs and return the current total.
_tcscpy(pszFilePart, pszWild);
hFind = FindFirstFile(pszSearchName, &wfd);
if (hFind != INVALID_HANDLE_VALUE)
{
do
{
if ((wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) &&
_tcsicmp (wfd.cFileName, TEXT("deleted")) &&
_tcscmp (wfd.cFileName, TEXT(".")) &&
_tcscmp (wfd.cFileName, TEXT("..")) &&
remaining >= _tcslen(wfd.cFileName))
{
_tcscpy(pszFilePart, wfd.cFileName);
DirUse = DoDu(pszSearchName);
ADDUSE(use, DirUse);
}
} while (FindNextFile(hFind, &wfd));
FindClose(hFind);
}
if (fSubtreeTotal)
TotPrint(&use, dir);
return(use);
}
void TotPrint (PUSESTAT puse, TCHAR *p)
{
static BOOL fFirst = TRUE;
TCHAR Buffer[MAX_PATH];
TCHAR *p1;
if (fFirst) {
// XXX,XXX,XXX,XXX XXX,XXX,XXX,XXX xx,xxx,xxx name
_tprintf( TEXT(" Used Allocated %s%s Files\n"),
fShowCompressed ? TEXT(" Compressed ") : TEXT(""),
// XXX,XXX,XXX,XXX
fShowDeleted ? TEXT(" Deleted ") : TEXT("")
// XXX,XXX,XXX,XXX
);
fFirst = FALSE;
}
// Disable processing so Middle Dot won't beep
// Middle Dot 0x2022 aliases to ^G when using Raster Fonts
SetConsoleMode(ghStdout, gdwOutputMode);
_tprintf(TEXT("%s "), FormatFileSize( puse->cchUsed, Buffer, 15 ));
_tprintf(TEXT("%s "), FormatFileSize( puse->cchAlloc, Buffer, 15 ));
if (fShowCompressed) {
_tprintf(TEXT("%s "), FormatFileSize( puse->cchCompressed, Buffer, 15 ));
}
if (fShowDeleted) {
_tprintf(TEXT("%s "), FormatFileSize( puse->cchDeleted, Buffer, 15 ));
}
_tprintf(TEXT("%s "), FormatFileSize( puse->cFile, Buffer, 10 ));
_tprintf(TEXT("%s"),p);
// Re-enable processing so newline works
SetConsoleMode(ghStdout, gdwOutputMode | ENABLE_PROCESSED_OUTPUT);
_tprintf(TEXT("\n"));
cDisp++;
}