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