LCOV - code coverage report
Current view: top level - src/common - rmtree.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 31 42 73.8 %
Date: 2025-01-18 04:15:08 Functions: 1 1 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * rmtree.c
       4             :  *
       5             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       6             :  * Portions Copyright (c) 1994, Regents of the University of California
       7             :  *
       8             :  * IDENTIFICATION
       9             :  *    src/common/rmtree.c
      10             :  *
      11             :  *-------------------------------------------------------------------------
      12             :  */
      13             : 
      14             : #ifndef FRONTEND
      15             : #include "postgres.h"
      16             : #else
      17             : #include "postgres_fe.h"
      18             : #endif
      19             : 
      20             : #include <unistd.h>
      21             : #include <sys/stat.h>
      22             : 
      23             : #include "common/file_utils.h"
      24             : 
      25             : #ifndef FRONTEND
      26             : #include "storage/fd.h"
      27             : #define pg_log_warning(...) elog(WARNING, __VA_ARGS__)
      28             : #define LOG_LEVEL WARNING
      29             : #define OPENDIR(x) AllocateDir(x)
      30             : #define CLOSEDIR(x) FreeDir(x)
      31             : #else
      32             : #include "common/logging.h"
      33             : #define LOG_LEVEL PG_LOG_WARNING
      34             : #define OPENDIR(x) opendir(x)
      35             : #define CLOSEDIR(x) closedir(x)
      36             : #endif
      37             : 
      38             : /*
      39             :  *  rmtree
      40             :  *
      41             :  *  Delete a directory tree recursively.
      42             :  *  Assumes path points to a valid directory.
      43             :  *  Deletes everything under path.
      44             :  *  If rmtopdir is true deletes the directory too.
      45             :  *  Returns true if successful, false if there was any problem.
      46             :  *  (The details of the problem are reported already, so caller
      47             :  *  doesn't really have to say anything more, but most do.)
      48             :  */
      49             : bool
      50        6670 : rmtree(const char *path, bool rmtopdir)
      51             : {
      52             :     char        pathbuf[MAXPGPATH];
      53             :     DIR        *dir;
      54             :     struct dirent *de;
      55        6670 :     bool        result = true;
      56        6670 :     size_t      dirnames_size = 0;
      57        6670 :     size_t      dirnames_capacity = 8;
      58             :     char      **dirnames;
      59             : 
      60        6670 :     dir = OPENDIR(path);
      61        6670 :     if (dir == NULL)
      62             :     {
      63           0 :         pg_log_warning("could not open directory \"%s\": %m", path);
      64           0 :         return false;
      65             :     }
      66             : 
      67        6670 :     dirnames = (char **) palloc(sizeof(char *) * dirnames_capacity);
      68             : 
      69      295542 :     while (errno = 0, (de = readdir(dir)))
      70             :     {
      71      288872 :         if (strcmp(de->d_name, ".") == 0 ||
      72      282202 :             strcmp(de->d_name, "..") == 0)
      73       13340 :             continue;
      74      275532 :         snprintf(pathbuf, sizeof(pathbuf), "%s/%s", path, de->d_name);
      75      275532 :         switch (get_dirent_type(pathbuf, de, false, LOG_LEVEL))
      76             :         {
      77           0 :             case PGFILETYPE_ERROR:
      78             :                 /* already logged, press on */
      79           0 :                 break;
      80        5258 :             case PGFILETYPE_DIR:
      81             : 
      82             :                 /*
      83             :                  * Defer recursion until after we've closed this directory, to
      84             :                  * avoid using more than one file descriptor at a time.
      85             :                  */
      86        5258 :                 if (dirnames_size == dirnames_capacity)
      87             :                 {
      88         370 :                     dirnames = repalloc(dirnames,
      89             :                                         sizeof(char *) * dirnames_capacity * 2);
      90         370 :                     dirnames_capacity *= 2;
      91             :                 }
      92        5258 :                 dirnames[dirnames_size++] = pstrdup(pathbuf);
      93        5258 :                 break;
      94      270274 :             default:
      95      270274 :                 if (unlink(pathbuf) != 0 && errno != ENOENT)
      96             :                 {
      97           0 :                     pg_log_warning("could not remove file \"%s\": %m", pathbuf);
      98           0 :                     result = false;
      99             :                 }
     100      270274 :                 break;
     101             :         }
     102             :     }
     103             : 
     104        6670 :     if (errno != 0)
     105             :     {
     106           0 :         pg_log_warning("could not read directory \"%s\": %m", path);
     107           0 :         result = false;
     108             :     }
     109             : 
     110        6670 :     CLOSEDIR(dir);
     111             : 
     112             :     /* Now recurse into the subdirectories we found. */
     113       11928 :     for (size_t i = 0; i < dirnames_size; ++i)
     114             :     {
     115        5258 :         if (!rmtree(dirnames[i], true))
     116           0 :             result = false;
     117        5258 :         pfree(dirnames[i]);
     118             :     }
     119             : 
     120        6670 :     if (rmtopdir)
     121             :     {
     122        6670 :         if (rmdir(path) != 0)
     123             :         {
     124           0 :             pg_log_warning("could not remove directory \"%s\": %m", path);
     125           0 :             result = false;
     126             :         }
     127             :     }
     128             : 
     129        6670 :     pfree(dirnames);
     130             : 
     131        6670 :     return result;
     132             : }

Generated by: LCOV version 1.14