add initial import

This commit is contained in:
Christien Rioux 2015-12-11 22:49:31 -08:00
parent ce8d316371
commit 7fbd70a2bb
5 changed files with 958 additions and 0 deletions

523
linecount.cpp Normal file
View File

@ -0,0 +1,523 @@
#include"linecount.h"
///////////////////////////// Platform specific
#ifdef _WIN32
// Windows
#define LCOPENFILE(name) CreateFile(name, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL)
#define LCCLOSEFILE(handle) CloseHandle(handle)
#define LCINVALIDHANDLE INVALID_HANDLE_VALUE
#define LCSETREALLASTERROR(err, errstr) { setLastError((err), (errstr)); }
typedef long long LCFILEOFFSET;
#elif defined (__unix__) || (defined (__APPLE__) && defined (__MACH__))
// POSIX
#include<sys/types.h>
#include<sys/stat.h>
#include<sys/fcntl.h>
#include<sys/mman.h>
#include<pthread.h>
#if (defined (__APPLE__) && defined (__MACH__)
#include <sys/sysctl.h>
typedef off_t LCFILEOFFSET;
#define MMAP mmap
#elif defined(__linux__)
typedef off64_t LCFILEOFFSET;
#define MMAP mmap64
#else
typedef off_t LCFILEOFFSET;
#define MMAP mmap
#endif
#define LCOPENFILE(name) open(name, O_RDONLY)
#define LCCLOSEFILE(handle) (close(handle) != -1)
#define LCINVALIDHANDLE -1
#define LCSETREALLASTERROR(err, errstr) { int __err = errno; setLastError(__err, strerror(__err)); }
#endif
///////////////////////////// Line Count Class
BEGIN_LINECOUNT_NAMESPACE;
struct LCTHREADCONTEXT
{
int thread_number;
CLineCount *m_this;
};
CLineCount::CLineCount(PARAMETERS *parameters)
{
// Set line count parameter defaults
int cpucount;
#ifdef _WIN32
SYSTEM_INFO sysinfo;
GetSystemInfo(&sysinfo);
cpucount = sysinfo.dwNumberOfProcessors;
#elif defined(__linux__)
cpucount = sysconf(_SC_NPROCESSORS_ONLN);
#elif (defined (__APPLE__) && defined (__MACH__))
intsize_t count_len = sizeof(cpucount);
sysctlbyname("hw.logicalcpu", &cpucount, &count_len, NULL, 0);
#else
cpucount = 1;
#endif
m_parameters.workercount = cpucount;
m_parameters.windowsize = (1024 * 1024 * 8) / cpucount;
// Override defaults if specified
if (parameters)
{
if (parameters->windowsize != -1)
{
m_parameters.windowsize = parameters->windowsize;
#ifdef _WIN32
// Window size must be multiple of allocation granularity. Round up.
SYSTEM_INFO si;
GetSystemInfo(&si);
m_parameters.windowsize += (si.dwAllocationGranularity - (m_parameters.windowsize % si.dwAllocationGranularity)) % si.dwAllocationGranularity;
#endif
}
if (parameters->workercount != -1)
{
m_parameters.workercount = parameters->workercount;
}
}
init();
}
CLineCount::~CLineCount()
{
if (m_auto_close && m_opened)
{
LCCLOSEFILE(m_fh);
}
}
void CLineCount::init(void)
{
m_lasterror = 0;
m_lasterrorstring = _T("");
m_opened = false;
m_auto_close = false;
m_fh = LCINVALIDHANDLE;
m_filesize = 0;
m_actual_thread_count = 0;
#ifdef _WIN32
m_filemapping = NULL;
#endif
m_threads.clear();
m_threadlinecounts.clear();
}
void CLineCount::setLastError(LCERROR lasterror, LCSTRING lasterrorstring)
{
m_lasterror = lasterror;
m_lasterrorstring = lasterrorstring;
}
LCERROR CLineCount::lastError() const
{
return m_lasterror;
}
LCSTRING CLineCount::lastErrorString() const
{
return m_lasterrorstring;
}
bool CLineCount::isOpened() const
{
return m_opened;
}
bool CLineCount::open(LCFILEHANDLE fhandle, bool auto_close)
{
if (m_opened)
{
setLastError(EEXIST, _T("file already opened"));
return false;
}
m_fh = fhandle;
m_opened = true;
m_auto_close = auto_close;
return true;
}
bool CLineCount::open(const TCHAR *filename)
{
if (m_opened)
{
setLastError(EEXIST, _T("file already opened"));
return false;
}
m_fh = LCOPENFILE(filename);
if (m_fh == LCINVALIDHANDLE)
{
LCSETREALLASTERROR(ENOENT, _T("file could not be opened"));
return false;
}
m_opened = true;
m_auto_close = true;
return true;
}
bool CLineCount::close()
{
if (!m_opened)
{
setLastError(EBADF, _T("file not opened"));
return false;
}
bool ok = true;
if (!LCCLOSEFILE(m_fh))
{
LCSETREALLASTERROR(EBADF, _T("unable to close file"));
ok = false;
}
init();
return ok;
}
#ifdef _WIN32
DWORD WINAPI threadProc(LPVOID ctx)
#else
void *threadProc(void *ctx)
#endif
{
LCTHREADCONTEXT *lctctx = (LCTHREADCONTEXT*)ctx;
return lctctx->m_this->countThread(lctctx->thread_number);
}
unsigned int CLineCount::countThread(int thread_number)
{
LCFILEOFFSET windowsize = (LCFILEOFFSET)m_parameters.windowsize;
LCFILEOFFSET startoffset = windowsize * (LCFILEOFFSET)thread_number;
LCFILEOFFSET stride = windowsize * m_actual_thread_count;
LCFILEOFFSET curoffset = startoffset;
LCFILEOFFSET lastmapsize = 0;
LCLINECOUNT count = 0;
void *mem = NULL;
while (curoffset < m_filesize)
{
if (m_thread_fail)
{
return -1;
}
// Get best file mapping window size
size_t mapsize = (size_t)min((m_filesize - curoffset), windowsize);
// Map view of file
#ifdef _WIN32
if (mem)
{
if (!UnmapViewOfFile(mem))
{
setLastError(EINVAL, _T("memory unmap failed"));
m_thread_fail = true;
return -1;
}
}
mem = MapViewOfFile(m_filemapping, FILE_MAP_READ, (DWORD)(curoffset >> 32), (DWORD)curoffset, (SIZE_T)mapsize);
#else
if (mem)
{
if(munmap(mem)!=0)
{
LCSETREALLASTERROR(EINVAL, _T("memory unmap failed"));
m_thread_fail = true;
return -1;
}
}
mem = MMAP(NULL, mapsize, PROT_READ, MAP_FILE | MAP_SHARED, m_fd, curoffset);
#endif
if (mem == NULL)
{
LCSETREALLASTERROR(EINVAL, _T("memory map failed"));
m_thread_fail = true;
return -1;
}
// Count newlines in buffer
LCFILEOFFSET windowoffset = 0;
size_t windowleft = mapsize;
char *ptr = (char *)mem;
while (windowleft > 0)
{
char *ptrnext = (char *)memchr(ptr, '\n', windowleft);
if (ptrnext)
{
ptrnext++;
count++;
windowleft -= (ptrnext - ptr);
ptr = ptrnext;
}
else
{
windowleft = 0;
}
}
// See if we need to account for end of file not ending with line terminator
if ((curoffset + mapsize) == m_filesize)
{
if (*((char *)mem + (mapsize - 1)) != '\n')
{
count++;
}
}
// Move to next buffer
curoffset += stride;
lastmapsize = mapsize;
}
// Clean up memory map
#ifdef _WIN32
if (mem)
{
if (!UnmapViewOfFile(mem))
{
setLastError(EINVAL, _T("memory unmap failed"));
m_thread_fail = true;
return -1;
}
}
#else
if (mem)
{
if (munmap(mem) != 0)
{
LCSETREALLASTERROR(EINVAL, _T("memory unmap failed"));
m_thread_fail = true;
return -1;
}
}
#endif
// Save count for this thread
m_threadlinecounts[thread_number] = count;
return 0;
}
bool CLineCount::createThread(int thread_number)
{
LCTHREADCONTEXT * ctx = new LCTHREADCONTEXT;
ctx->m_this = this;
ctx->thread_number = thread_number;
#ifdef _WIN32
HANDLE hThread = CreateThread(NULL, 0, threadProc, ctx, 0, NULL);
if(!hThread)
{
return false;
}
#else
pthread_t hThread;
int ret = pthread_create(&hThread, NULL, threadProc, ctx);
if (ret != 0)
{
return false;
}
#endif
m_threads[thread_number] = hThread;
return true;
}
bool CLineCount::countLines(LCLINECOUNT & linecount)
{
// Determine file size
#ifdef _WIN32
LARGE_INTEGER li;
if (!GetFileSizeEx(m_fh, &li))
{
LCSETREALLASTERROR(EBADF, _T("unable to get file size"));
return false;
}
m_filesize = li.QuadPart;
#else
struct stat64 statbuf;
if(fstat64(m_fh,&statbuf)!=0)
{
LCSETREALLASTERROR(EBADF, _T("unable to get file size"));
return false;
}
m_filesize = statbuf.st_size;
#endif
// Exit now for empty files
if (m_filesize == 0)
{
linecount = 0;
return true;
}
// Figure out actual thread count
LCFILEOFFSET windowcount = (m_filesize + (m_parameters.windowsize - 1)) / m_parameters.windowsize;
if (windowcount < (LCFILEOFFSET) m_parameters.workercount)
{
m_actual_thread_count = (int)windowcount;
}
else
{
m_actual_thread_count = m_parameters.workercount;
}
#ifdef _WIN32
// Prepare file mapping
m_filemapping = CreateFileMapping(m_fh, NULL, PAGE_READONLY, 0, 0, NULL);
#endif
// Spin up threads
m_threads.resize(m_actual_thread_count);
m_threadlinecounts.resize(m_actual_thread_count);
m_thread_fail = false;
for (int i = 0; i < m_actual_thread_count; i++)
{
if (!createThread(i))
{
setLastError(ECHILD, _T("failed to create counting thread"));
m_thread_fail = true;
}
}
// Wait for threads to complete
int complete = 0;
int errors = 0;
for (int i = 0; i < m_actual_thread_count; i++)
{
bool success = false;
if (m_threads[i] != NULL)
{
#ifdef _WIN32
success = (WaitForSingleObject(m_threads[i], INFINITE) == WAIT_OBJECT_0);
#else
success = pthread_join(m_threads[i]) == 0;
#endif
}
if (success)
{
complete++;
}
else
{
errors++;
}
}
#ifdef _WIN32
// Clean up file mapping
CloseHandle(m_filemapping);
#endif
if (m_thread_fail)
{
return false;
}
if (complete != m_parameters.workercount)
{
setLastError(ECHILD, _T("thread join failed"));
return false;
}
// Sum up thread line counts and return
linecount = 0;
for (int i = 0; i < m_actual_thread_count; i++)
{
linecount += m_threadlinecounts[i];
}
return true;
}
// Static helpers
LCLINECOUNT CLineCount::LineCount(LCFILEHANDLE fhandle, LCERROR * error, LCSTRING *errorstring)
{
CLineCount lc;
if (!lc.open(fhandle))
{
if (error)
{
*error = lc.lastError();
}
if (errorstring)
{
*errorstring = lc.lastErrorString();
}
return -1;
}
LCLINECOUNT count;
if (!lc.countLines(count))
{
if (error)
{
*error = lc.lastError();
}
if (errorstring)
{
*errorstring = lc.lastErrorString();
}
return -1;
}
*error = 0;
*errorstring = _T("");
return count;
}
LCLINECOUNT CLineCount::LineCount(const TCHAR *filename, LCERROR * error, LCSTRING *errorstring)
{
CLineCount lc;
if (!lc.open(filename))
{
if (error)
{
*error = lc.lastError();
}
if (errorstring)
{
*errorstring = lc.lastErrorString();
}
return -1;
}
LCLINECOUNT count;
if (!lc.countLines(count))
{
if (error)
{
*error = lc.lastError();
}
if (errorstring)
{
*errorstring = lc.lastErrorString();
}
return -1;
}
*error = 0;
*errorstring = _T("");
return count;
}
END_LINECOUNT_NAMESPACE;

128
linecount.h Normal file
View File

@ -0,0 +1,128 @@
#ifndef __INC_LINECOUNT_H
#define __INC_LINECOUNT_H
///////////////////////////////////////////// Headers
////////////// Platform independent
#include<string>
#include<vector>
#include<errno.h>
#define BEGIN_LINECOUNT_NAMESPACE namespace LineCount {
#define END_LINECOUNT_NAMESPACE }
////////////// Platform specific
#ifdef _WIN32 // Windows
#include<Windows.h>
#include<tchar.h>
#elif defined (__unix__) || (defined (__APPLE__) && defined (__MACH__)) // POSIX
#include<unistd.h>
#define _T(x) x
#define TCHAR char
#endif
///////////////////////////////////////////// Line Count Class
BEGIN_LINECOUNT_NAMESPACE;
////////////// Platform specific
#ifdef _WIN32 // Windows
#ifdef _UNICODE
typedef std::wstring LCSTRING;
#else
typedef std::string LCSTRING;
#endif
typedef HANDLE LCFILEHANDLE;
typedef errno_t LCERROR;
typedef long long LCFILEOFFSET;
typedef LCFILEOFFSET LCLINECOUNT;
#define LCLINECOUNTFMT "%I64d"
#elif defined (__unix__) || (defined (__APPLE__) && defined (__MACH__)) // POSIX
typedef std::string LCSTRING;
typedef int LCFILEHANDLE;
typedef errno_t LCERROR;
#if (defined (__APPLE__) && defined (__MACH__)
typedef off_t LCFILEOFFSET;
#define LCLINECOUNTFMT "%lld"
#elif defined(__linux__)
typedef off64_t LCFILEOFFSET;
#define LCLINECOUNTFMT "%lld"
#else
typedef off_t LCFILEOFFSET;
#define LCLINECOUNTFMT "%d"
#endif
typedef LCFILEOFFSET LCLINECOUNT;
#endif
class CLineCount
{
public:
struct PARAMETERS
{
size_t windowsize;
int workercount;
};
private:
bool m_opened;
bool m_auto_close;
LCFILEHANDLE m_fh;
LCERROR m_lasterror;
LCSTRING m_lasterrorstring;
LCFILEOFFSET m_filesize;
PARAMETERS m_parameters;
int m_actual_thread_count;
#ifdef _WIN32
std::vector<HANDLE> m_threads;
HANDLE m_filemapping;
#else
std::vector<pthread_t> m_threads;
#endif
std::vector<LCLINECOUNT> m_threadlinecounts;
bool m_thread_fail;
private:
void setLastError(LCERROR error, LCSTRING lasterrorstring);
void init();
bool createThread(int thread_number);
#ifdef _WIN32
friend DWORD WINAPI threadProc(LPVOID ctx);
#else
friend void *threadProc(void *ctx);
#endif
unsigned int countThread(int thread_number);
public:
CLineCount(PARAMETERS *parameters=NULL);
~CLineCount();
bool isOpened() const;
LCERROR lastError() const;
LCSTRING lastErrorString() const;
bool open(LCFILEHANDLE fhandle, bool auto_close = false);
bool open(const TCHAR * filename);
bool close();
bool countLines(LCLINECOUNT &linecount);
public:
// Static utility functions
static LCLINECOUNT LineCount(LCFILEHANDLE fhandle, LCERROR * error = NULL, LCSTRING * errorstring = NULL);
static LCLINECOUNT LineCount(const TCHAR *filename, LCERROR * error = NULL, LCSTRING * errorstring = NULL);
};
END_LINECOUNT_NAMESPACE;
#endif

28
linecount.sln Normal file
View File

@ -0,0 +1,28 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio 2013
VisualStudioVersion = 12.0.31101.0
MinimumVisualStudioVersion = 10.0.40219.1
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "linecount", "linecount.vcxproj", "{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Win32 = Debug|Win32
Debug|x64 = Debug|x64
Release|Win32 = Release|Win32
Release|x64 = Release|x64
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}.Debug|Win32.ActiveCfg = Debug|Win32
{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}.Debug|Win32.Build.0 = Debug|Win32
{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}.Debug|x64.ActiveCfg = Debug|x64
{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}.Debug|x64.Build.0 = Debug|x64
{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}.Release|Win32.ActiveCfg = Release|Win32
{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}.Release|Win32.Build.0 = Release|Win32
{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}.Release|x64.ActiveCfg = Release|x64
{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}.Release|x64.Build.0 = Release|x64
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
EndGlobal

151
linecount.vcxproj Normal file
View File

@ -0,0 +1,151 @@
<?xml version="1.0" encoding="utf-8"?>
<Project DefaultTargets="Build" ToolsVersion="12.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<ItemGroup Label="ProjectConfigurations">
<ProjectConfiguration Include="Debug|Win32">
<Configuration>Debug</Configuration>
<Platform>Win32</Platform>
</ProjectConfiguration>
<ProjectConfiguration Include="Debug|x64">
<Configuration>Debug</Configuration>
<Platform>x64</Platform>
</ProjectConfiguration>
<ProjectConfiguration Include="Release|Win32">
<Configuration>Release</Configuration>
<Platform>Win32</Platform>
</ProjectConfiguration>
<ProjectConfiguration Include="Release|x64">
<Configuration>Release</Configuration>
<Platform>x64</Platform>
</ProjectConfiguration>
</ItemGroup>
<PropertyGroup Label="Globals">
<ProjectGuid>{E5D80D5A-FDE6-44FE-9E97-DF3D470DD5E9}</ProjectGuid>
<Keyword>Win32Proj</Keyword>
<RootNamespace>linecount</RootNamespace>
</PropertyGroup>
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration">
<ConfigurationType>Application</ConfigurationType>
<UseDebugLibraries>true</UseDebugLibraries>
<PlatformToolset>v120</PlatformToolset>
<CharacterSet>Unicode</CharacterSet>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'" Label="Configuration">
<ConfigurationType>Application</ConfigurationType>
<UseDebugLibraries>true</UseDebugLibraries>
<PlatformToolset>v120</PlatformToolset>
<CharacterSet>Unicode</CharacterSet>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration">
<ConfigurationType>Application</ConfigurationType>
<UseDebugLibraries>false</UseDebugLibraries>
<PlatformToolset>v120</PlatformToolset>
<WholeProgramOptimization>true</WholeProgramOptimization>
<CharacterSet>Unicode</CharacterSet>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'" Label="Configuration">
<ConfigurationType>Application</ConfigurationType>
<UseDebugLibraries>false</UseDebugLibraries>
<PlatformToolset>v120</PlatformToolset>
<WholeProgramOptimization>true</WholeProgramOptimization>
<CharacterSet>Unicode</CharacterSet>
</PropertyGroup>
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
<ImportGroup Label="ExtensionSettings">
</ImportGroup>
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
</ImportGroup>
<ImportGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'" Label="PropertySheets">
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
</ImportGroup>
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
</ImportGroup>
<ImportGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'" Label="PropertySheets">
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
</ImportGroup>
<PropertyGroup Label="UserMacros" />
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
<LinkIncremental>true</LinkIncremental>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
<LinkIncremental>true</LinkIncremental>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
<LinkIncremental>false</LinkIncremental>
</PropertyGroup>
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
<LinkIncremental>false</LinkIncremental>
</PropertyGroup>
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
<ClCompile>
<PrecompiledHeader>
</PrecompiledHeader>
<WarningLevel>Level3</WarningLevel>
<Optimization>Disabled</Optimization>
<PreprocessorDefinitions>WIN32;_DEBUG;_CONSOLE;_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions>
</ClCompile>
<Link>
<SubSystem>Console</SubSystem>
<GenerateDebugInformation>true</GenerateDebugInformation>
</Link>
</ItemDefinitionGroup>
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
<ClCompile>
<PrecompiledHeader>
</PrecompiledHeader>
<WarningLevel>Level3</WarningLevel>
<Optimization>Disabled</Optimization>
<PreprocessorDefinitions>WIN32;_DEBUG;_CONSOLE;_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions>
</ClCompile>
<Link>
<SubSystem>Console</SubSystem>
<GenerateDebugInformation>true</GenerateDebugInformation>
</Link>
</ItemDefinitionGroup>
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
<ClCompile>
<WarningLevel>Level3</WarningLevel>
<PrecompiledHeader>
</PrecompiledHeader>
<Optimization>MaxSpeed</Optimization>
<FunctionLevelLinking>true</FunctionLevelLinking>
<IntrinsicFunctions>true</IntrinsicFunctions>
<PreprocessorDefinitions>WIN32;NDEBUG;_CONSOLE;_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions>
</ClCompile>
<Link>
<SubSystem>Console</SubSystem>
<GenerateDebugInformation>true</GenerateDebugInformation>
<EnableCOMDATFolding>true</EnableCOMDATFolding>
<OptimizeReferences>true</OptimizeReferences>
</Link>
</ItemDefinitionGroup>
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
<ClCompile>
<WarningLevel>Level3</WarningLevel>
<PrecompiledHeader>
</PrecompiledHeader>
<Optimization>MaxSpeed</Optimization>
<FunctionLevelLinking>true</FunctionLevelLinking>
<IntrinsicFunctions>true</IntrinsicFunctions>
<PreprocessorDefinitions>WIN32;NDEBUG;_CONSOLE;_LIB;%(PreprocessorDefinitions)</PreprocessorDefinitions>
</ClCompile>
<Link>
<SubSystem>Console</SubSystem>
<GenerateDebugInformation>true</GenerateDebugInformation>
<EnableCOMDATFolding>true</EnableCOMDATFolding>
<OptimizeReferences>true</OptimizeReferences>
</Link>
</ItemDefinitionGroup>
<ItemGroup>
<ClCompile Include="linecount.cpp" />
<ClCompile Include="main.cpp" />
</ItemGroup>
<ItemGroup>
<ClInclude Include="linecount.h" />
</ItemGroup>
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
<ImportGroup Label="ExtensionTargets">
</ImportGroup>
</Project>

128
main.cpp Normal file
View File

@ -0,0 +1,128 @@
#include"linecount.h"
#ifdef _WIN32
#include<tchar.h>
#else
#define _tprintf printf
#define _ftprintf fprintf
#define _tcscmp strcmp
#define _ttoi atoi
#define _tcstoui64 strtoull
#define _T(x) x
#endif
using namespace LineCount;
#if defined(WIN32) && defined(_UNICODE)
int wmain(int argc, TCHAR **argv)
#else
int main(int argc, char **argv)
#endif
{
// Parse parameters
int arg = 1;
int posparam = 0;
CLineCount::PARAMETERS params;
params.windowsize = -1;
params.workercount = -1;
TCHAR *filename = NULL;
while (arg < argc)
{
if (_tcscmp(argv[arg], _T("--windowsize")) == 0)
{
arg++;
if (arg == argc)
{
_ftprintf(stderr, _T("missing argument to --windowsize"));
return 1;
}
_TCHAR *wsstr = argv[arg];
// Check for size multipliers
size_t multiplier = 1;
_TCHAR *lastchar = wsstr + (_tcslen(wsstr) - 1);
if (*lastchar == _T('k') || *lastchar == _T('K'))
{
multiplier = 1024;
lastchar = 0;
}
else if (*lastchar == _T('m') || *lastchar == _T('M'))
{
multiplier = 1024 * 1024;
lastchar = 0;
}
else if (*lastchar == _T('g') || *lastchar == _T('G'))
{
multiplier = 1024 * 1024 * 1024;
lastchar = 0;
}
_TCHAR *endptr;
params.windowsize = ((size_t)_tcstoui64(argv[arg], &endptr, 10)) * multiplier;
}
else if (_tcscmp(argv[arg], _T("--workercount")) == 0)
{
arg++;
if (arg == argc)
{
_ftprintf(stderr, _T("missing argument to --workercount"));
return 1;
}
params.workercount = _ttoi(argv[arg]);
}
else
{
if (posparam == 0)
{
filename = argv[arg];
}
else
{
_ftprintf(stderr, _T("too many arguments"));
return 1;
}
posparam++;
}
arg++;
}
if (posparam != 1)
{
_ftprintf(stderr, _T("missing required argument"));
return 1;
}
// Create line count class
CLineCount lc(&params);
if (!lc.open(filename))
{
LCERROR err = lc.lastError();
LCSTRING errstr = lc.lastErrorString();
_ftprintf(stderr, _T("error (%d): %s\n"), err, errstr.c_str());
return err;
}
// Count lines
LCLINECOUNT count;
if (!lc.countLines(count))
{
LCERROR err = lc.lastError();
LCSTRING errstr = lc.lastErrorString();
_ftprintf(stderr, _T("error (%d): %s\n"), err, errstr.c_str());
return err;
}
// Display output
_tprintf(_T(LCLINECOUNTFMT _T("\n")), count);
return 0;
}