simplify_path.c 1.6 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364
  1. /* vi: set sw=4 ts=4: */
  2. /*
  3. * bb_simplify_path implementation for busybox
  4. *
  5. * Copyright (C) 2001 Manuel Novoa III <mjn3@codepoet.org>
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program; if not, write to the Free Software
  19. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  20. *
  21. */
  22. #include <stdlib.h>
  23. #include "libbb.h"
  24. char *bb_simplify_path(const char *path)
  25. {
  26. char *s, *start, *p;
  27. if (path[0] == '/')
  28. start = bb_xstrdup(path);
  29. else {
  30. s = xgetcwd(NULL);
  31. start = concat_path_file(s, path);
  32. free(s);
  33. }
  34. p = s = start;
  35. do {
  36. if (*p == '/') {
  37. if (*s == '/') { /* skip duplicate (or initial) slash */
  38. continue;
  39. } else if (*s == '.') {
  40. if (s[1] == '/' || s[1] == 0) { /* remove extra '.' */
  41. continue;
  42. } else if ((s[1] == '.') && (s[2] == '/' || s[2] == 0)) {
  43. ++s;
  44. if (p > start) {
  45. while (*--p != '/'); /* omit previous dir */
  46. }
  47. continue;
  48. }
  49. }
  50. }
  51. *++p = *s;
  52. } while (*++s);
  53. if ((p == start) || (*p != '/')) { /* not a trailing slash */
  54. ++p; /* so keep last character */
  55. }
  56. *p = 0;
  57. return start;
  58. }