|  | // 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 <errno.h> | 
|  | #include <libgen.h> | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  | #include <unistd.h> | 
|  |  | 
|  | // 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 <path>\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; | 
|  | } |