185 lines
3.8 KiB
C++
185 lines
3.8 KiB
C++
/*
|
|
* Copyright (C) 2005 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
//
|
|
// Miscellaneous utility functions.
|
|
//
|
|
#include <utils/misc.h>
|
|
|
|
#include <sys/stat.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <assert.h>
|
|
#include <stdio.h>
|
|
|
|
using namespace android;
|
|
|
|
namespace android {
|
|
|
|
/*
|
|
* Like strdup(), but uses C++ "new" operator instead of malloc.
|
|
*/
|
|
char* strdupNew(const char* str)
|
|
{
|
|
char* newStr;
|
|
int len;
|
|
|
|
if (str == NULL)
|
|
return NULL;
|
|
|
|
len = strlen(str);
|
|
newStr = new char[len+1];
|
|
memcpy(newStr, str, len+1);
|
|
|
|
return newStr;
|
|
}
|
|
|
|
/*
|
|
* Concatenate an argument vector.
|
|
*/
|
|
char* concatArgv(int argc, const char* const argv[])
|
|
{
|
|
char* newStr = NULL;
|
|
int len, totalLen, posn, idx;
|
|
|
|
/*
|
|
* First, figure out the total length.
|
|
*/
|
|
totalLen = idx = 0;
|
|
while (1) {
|
|
if (idx == argc || argv[idx] == NULL)
|
|
break;
|
|
if (idx)
|
|
totalLen++; // leave a space between args
|
|
totalLen += strlen(argv[idx]);
|
|
idx++;
|
|
}
|
|
|
|
/*
|
|
* Alloc the string.
|
|
*/
|
|
newStr = new char[totalLen +1];
|
|
if (newStr == NULL)
|
|
return NULL;
|
|
|
|
/*
|
|
* Finally, allocate the string and copy data over.
|
|
*/
|
|
idx = posn = 0;
|
|
while (1) {
|
|
if (idx == argc || argv[idx] == NULL)
|
|
break;
|
|
if (idx)
|
|
newStr[posn++] = ' ';
|
|
|
|
len = strlen(argv[idx]);
|
|
memcpy(&newStr[posn], argv[idx], len);
|
|
posn += len;
|
|
|
|
idx++;
|
|
}
|
|
|
|
assert(posn == totalLen);
|
|
newStr[posn] = '\0';
|
|
|
|
return newStr;
|
|
}
|
|
|
|
/*
|
|
* Count the #of args in an argument vector. Don't count the final NULL.
|
|
*/
|
|
int countArgv(const char* const argv[])
|
|
{
|
|
int count = 0;
|
|
|
|
while (argv[count] != NULL)
|
|
count++;
|
|
|
|
return count;
|
|
}
|
|
|
|
|
|
#include <stdio.h>
|
|
/*
|
|
* Get a file's type.
|
|
*/
|
|
FileType getFileType(const char* fileName)
|
|
{
|
|
struct stat sb;
|
|
|
|
if (stat(fileName, &sb) < 0) {
|
|
if (errno == ENOENT || errno == ENOTDIR)
|
|
return kFileTypeNonexistent;
|
|
else {
|
|
fprintf(stderr, "getFileType got errno=%d on '%s'\n",
|
|
errno, fileName);
|
|
return kFileTypeUnknown;
|
|
}
|
|
} else {
|
|
if (S_ISREG(sb.st_mode))
|
|
return kFileTypeRegular;
|
|
else if (S_ISDIR(sb.st_mode))
|
|
return kFileTypeDirectory;
|
|
else if (S_ISCHR(sb.st_mode))
|
|
return kFileTypeCharDev;
|
|
else if (S_ISBLK(sb.st_mode))
|
|
return kFileTypeBlockDev;
|
|
else if (S_ISFIFO(sb.st_mode))
|
|
return kFileTypeFifo;
|
|
#ifdef HAVE_SYMLINKS
|
|
else if (S_ISLNK(sb.st_mode))
|
|
return kFileTypeSymlink;
|
|
else if (S_ISSOCK(sb.st_mode))
|
|
return kFileTypeSocket;
|
|
#endif
|
|
else
|
|
return kFileTypeUnknown;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Get a file's modification date.
|
|
*/
|
|
time_t getFileModDate(const char* fileName)
|
|
{
|
|
struct stat sb;
|
|
|
|
if (stat(fileName, &sb) < 0)
|
|
return (time_t) -1;
|
|
|
|
return sb.st_mtime;
|
|
}
|
|
|
|
/*
|
|
* Round up to the next highest power of 2.
|
|
*
|
|
* Found on http://graphics.stanford.edu/~seander/bithacks.html.
|
|
*/
|
|
unsigned int roundUpPower2(unsigned int val)
|
|
{
|
|
val--;
|
|
val |= val >> 1;
|
|
val |= val >> 2;
|
|
val |= val >> 4;
|
|
val |= val >> 8;
|
|
val |= val >> 16;
|
|
val++;
|
|
|
|
return val;
|
|
}
|
|
|
|
}; // namespace android
|
|
|