/*
* 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