// Copyright 2015 The Bazel Authors. All rights reserved. // // 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. // // realpath.c // // A simple implementation of realpath for Mac OS X. // This implementation follows gnu/linux conventions and allows the last // component to not exist: // http://www.gnu.org/software/coreutils/manual/html_node/realpath-invocation.html // Debian requires all components to exist. // #include #include #include #include #include #include // Print a simple error message and exit. static void PrintError(const char *argv[]) { fprintf(stderr, "%s: %s\n", argv[1], strerror(errno)); exit(1); } // Concatenate two paths together adding a '/' if appropriate. // Returned pointer is owned by client and should be freed. static char *JoinPaths(const char *path1, const char* path2) { size_t len1 = strlen(path1); size_t len2 = strlen(path2); // +1 for '/' and +1 for '\0' size_t totalSize = len1 + 1 + len2 + 1; char *outPath = malloc(totalSize); if (outPath == NULL) { return NULL; } strlcpy(outPath, path1, totalSize); if (len1 > 0 && len2 > 0) { if (path1[len1 - 1] != '/' && path2[0] != '/') { strlcat(outPath, "/", totalSize); } } strlcat(outPath, path2, totalSize); return outPath; } // Since this is a simple utility that quits immediately, we are not worrying // about making the code more complex by freeing up any memory allocations. int main(int argc, const char *argv[]) { if (argc != 2) { fprintf(stderr, "realpath \n"); return 1; } const char *path = argv[1]; char *goodPath = realpath(path, NULL); if (goodPath == NULL) { if ((errno != ENOENT) || (strlen(path) == 0)) { PrintError(argv); } // If only the last element is missing, then call realpath on the parent // dir and append the basename back onto it. // Technically the strdup is not required on Mac OS X, but this // keeps things compatible with other basename/dirname implementations // that do require a string they can modify. char *dirCopy = strdup(path); char *baseCopy = strdup(path); if (dirCopy == NULL || baseCopy == NULL) { PrintError(argv); } char *dir = dirname(dirCopy); if (dir == NULL) { PrintError(argv); } char *base = basename(baseCopy); if (base == NULL) { PrintError(argv); } char *realdir = realpath(dir, NULL); if (realdir == NULL) { PrintError(argv); } goodPath = JoinPaths(realdir, base); if (goodPath == NULL) { PrintError(argv); } } fprintf(stdout, "%s\n", goodPath); return 0; }