LCOV - code coverage report
Current view: top level - lib/util - util_file.c (source / functions) Hit Total Coverage
Test: coverage report for v4-17-test 1498b464 Lines: 129 228 56.6 %
Date: 2024-06-13 04:01:37 Functions: 10 14 71.4 %

          Line data    Source code
       1             : /*
       2             :  * Unix SMB/CIFS implementation.
       3             :  * SMB parameters and setup
       4             :  * Copyright (C) Andrew Tridgell 1992-1998 Modified by Jeremy Allison 1995.
       5             :  *
       6             :  * Added afdgets() Jelmer Vernooij 2005
       7             :  *
       8             :  * This program is free software; you can redistribute it and/or modify it under
       9             :  * the terms of the GNU General Public License as published by the Free
      10             :  * Software Foundation; either version 3 of the License, or (at your option)
      11             :  * any later version.
      12             :  *
      13             :  * This program is distributed in the hope that it will be useful, but WITHOUT
      14             :  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
      15             :  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
      16             :  * more details.
      17             :  *
      18             :  * You should have received a copy of the GNU General Public License along with
      19             :  * this program; if not, see <http://www.gnu.org/licenses/>.
      20             :  */
      21             : 
      22             : #include "replace.h"
      23             : #include "system/shmem.h"
      24             : #include "system/filesys.h"
      25             : #include <talloc.h>
      26             : #include "lib/util/samba_util.h"
      27             : #include "lib/util/sys_popen.h"
      28             : #include "lib/util/sys_rw.h"
      29             : #include "lib/util/debug.h"
      30             : 
      31             : /**
      32             :  * Read one line (data until next newline or eof) and allocate it
      33             :  */
      34        7108 : _PUBLIC_ char *afdgets(int fd, TALLOC_CTX *mem_ctx, size_t hint)
      35             : {
      36        7108 :         char *data = NULL;
      37        7108 :         ssize_t alloc_size = 0, offset = 0, ret;
      38             :         int p;
      39             : 
      40        7108 :         if (hint <= 0) hint = 0x100;
      41             : 
      42             :         do {
      43        7108 :                 alloc_size += hint;
      44             : 
      45        7108 :                 data = talloc_realloc(mem_ctx, data, char, alloc_size);
      46             : 
      47        7108 :                 if (!data)
      48           0 :                         return NULL;
      49             : 
      50        7108 :                 ret = read(fd, data + offset, hint);
      51             : 
      52        7108 :                 if (ret == 0) {
      53         147 :                         return NULL;
      54             :                 }
      55             : 
      56        6961 :                 if (ret == -1) {
      57           0 :                         talloc_free(data);
      58           0 :                         return NULL;
      59             :                 }
      60             : 
      61             :                 /* Find newline */
      62      164235 :                 for (p = 0; p < ret; p++) {
      63      164235 :                         if (data[offset + p] == '\n')
      64        6961 :                                 break;
      65             :                 }
      66             : 
      67        6961 :                 if (p < ret) {
      68        6961 :                         data[offset + p] = '\0';
      69             : 
      70             :                         /* Go back to position of newline */
      71        6961 :                         lseek(fd, p - ret + 1, SEEK_CUR);
      72        6961 :                         return data;
      73             :                 }
      74             : 
      75           0 :                 offset += ret;
      76             : 
      77           0 :         } while ((size_t)ret == hint);
      78             : 
      79           0 :         data[offset] = '\0';
      80             : 
      81           0 :         return data;
      82             : }
      83             : 
      84         232 : char *fgets_slash(TALLOC_CTX *mem_ctx, char *s2, size_t maxlen, FILE *f)
      85             : {
      86         232 :         char *s = s2;
      87         232 :         size_t len = 0;
      88             :         int c;
      89         232 :         bool start_of_line = true;
      90             : 
      91         232 :         if (feof(f)) {
      92           0 :                 return NULL;
      93             :         }
      94             : 
      95         232 :         if (maxlen < 2) {
      96           0 :                 return NULL;
      97             :         }
      98             : 
      99         232 :         if (s2 == NULL) {
     100           0 :                 maxlen = MIN(maxlen,8);
     101           0 :                 s = talloc_array(mem_ctx, char, maxlen);
     102             :         }
     103             : 
     104         232 :         if (s == NULL) {
     105           0 :                 return NULL;
     106             :         }
     107             : 
     108         232 :         *s = 0;
     109             : 
     110        4142 :         while (len < maxlen-1) {
     111        4026 :                 c = getc(f);
     112        4026 :                 switch (c)
     113             :                 {
     114           0 :                     case '\r':
     115           0 :                             break;
     116         194 :                     case '\n':
     117         291 :                             while (len > 0 && s[len-1] == ' ') {
     118           0 :                                     s[--len] = 0;
     119             :                             }
     120         194 :                             if (len > 0 && s[len-1] == '\\') {
     121           0 :                                     s[--len] = 0;
     122           0 :                                     start_of_line = true;
     123           0 :                                     break;
     124             :                             }
     125         194 :                             return s;
     126          38 :                     case EOF:
     127          38 :                             if (len <= 0 && (s2 == NULL)) {
     128           0 :                                     TALLOC_FREE(s);
     129             :                             }
     130          38 :                             return (len>0) ? s : NULL;
     131         308 :                     case ' ':
     132         308 :                             if (start_of_line) {
     133           0 :                                     break;
     134             :                             }
     135             : 
     136             :                             FALL_THROUGH;
     137             :                     default:
     138        3794 :                             start_of_line = false;
     139        3794 :                             s[len++] = c;
     140        3794 :                             s[len] = 0;
     141             :                 }
     142        3794 :                 if ((s2 == NULL) && (len > maxlen-3)) {
     143             :                         size_t m;
     144             :                         char *t;
     145             : 
     146           0 :                         m = maxlen * 2;
     147           0 :                         if (m < maxlen) {
     148           0 :                                 DBG_ERR("length overflow");
     149           0 :                                 TALLOC_FREE(s);
     150           0 :                                 return NULL;
     151             :                         }
     152           0 :                         maxlen = m;
     153             : 
     154           0 :                         t = talloc_realloc(mem_ctx, s, char, maxlen);
     155           0 :                         if (t == NULL) {
     156           0 :                                 DBG_ERR("failed to expand buffer!\n");
     157           0 :                                 TALLOC_FREE(s);
     158           0 :                                 return NULL;
     159             :                         }
     160             : 
     161           0 :                         s = t;
     162             :                 }
     163             :         }
     164             : 
     165           0 :         return s;
     166             : }
     167             : 
     168             : /**
     169             : load a file into memory from a fd.
     170             : **/
     171          67 : _PUBLIC_ char *fd_load(int fd, size_t *psize, size_t maxsize, TALLOC_CTX *mem_ctx)
     172             : {
     173             :         FILE *file;
     174          67 :         char *p = NULL;
     175          67 :         size_t size = 0;
     176          67 :         size_t chunk = 1024;
     177             :         int err;
     178             :         int fd_dup;
     179             : 
     180          67 :         if (maxsize == 0) {
     181          67 :                 maxsize = SIZE_MAX;
     182             :         }
     183             : 
     184          67 :         fd_dup = dup(fd);
     185          67 :         if (fd_dup == -1) {
     186           0 :                 return NULL;
     187             :         }
     188             : 
     189          67 :         file = fdopen(fd_dup, "r");
     190          67 :         if (file == NULL) {
     191           0 :                 close(fd_dup);
     192           0 :                 return NULL;
     193             :         }
     194             : 
     195         161 :         while (size < maxsize) {
     196             :                 size_t newbufsize;
     197             :                 size_t nread;
     198             : 
     199         115 :                 chunk = MIN(chunk, (maxsize - size));
     200             : 
     201         115 :                 newbufsize = size + (chunk+1); /* chunk+1 can't overflow */
     202         115 :                 if (newbufsize < size) {
     203           0 :                         goto fail; /* overflow */
     204             :                 }
     205             : 
     206         115 :                 p = talloc_realloc(mem_ctx, p, char, newbufsize);
     207         115 :                 if (p == NULL) {
     208           0 :                         goto fail;
     209             :                 }
     210             : 
     211         115 :                 nread = fread(p+size, 1, chunk, file);
     212         115 :                 size += nread;
     213             : 
     214         115 :                 if (nread != chunk) {
     215          67 :                         break;
     216             :                 }
     217             :         }
     218             : 
     219          67 :         err = ferror(file);
     220          67 :         if (err != 0) {
     221           0 :                 goto fail;
     222             :         }
     223             : 
     224          67 :         p[size] = '\0';
     225             : 
     226          67 :         if (psize != NULL) {
     227          67 :                 *psize = size;
     228             :         }
     229             : 
     230          67 :         fclose(file);
     231          67 :         return p;
     232             : 
     233           0 : fail:
     234           0 :         TALLOC_FREE(p);
     235           0 :         fclose(file);
     236           0 :         return NULL;
     237             : }
     238             : 
     239             : /**
     240             : load a file into memory
     241             : **/
     242         159 : _PUBLIC_ char *file_load(const char *fname, size_t *size, size_t maxsize, TALLOC_CTX *mem_ctx)
     243             : {
     244             :         int fd;
     245             :         char *p;
     246             : 
     247         159 :         if (!fname || !*fname) return NULL;
     248             : 
     249         159 :         fd = open(fname,O_RDONLY);
     250         159 :         if (fd == -1) return NULL;
     251             : 
     252          67 :         p = fd_load(fd, size, maxsize, mem_ctx);
     253             : 
     254          67 :         close(fd);
     255             : 
     256          67 :         return p;
     257             : }
     258             : 
     259             : /**
     260             : parse a buffer into lines
     261             : 'p' will be freed on error, and otherwise will be made a child of the returned array
     262             : **/
     263         293 : static char **file_lines_parse_internal(char *p, size_t size, int *numlines, TALLOC_CTX *mem_ctx)
     264             : {
     265             :         unsigned int i;
     266             :         char *s, **ret;
     267             : 
     268         293 :         if (!p) return NULL;
     269             : 
     270       57935 :         for (s = p, i=0; s < p+size; s++) {
     271       57642 :                 if (s[0] == '\n') i++;
     272             :         }
     273             : 
     274         293 :         ret = talloc_zero_array(mem_ctx, char *, i+2);
     275         293 :         if (!ret) {
     276           0 :                 talloc_free(p);
     277           0 :                 return NULL;
     278             :         }
     279             : 
     280         293 :         talloc_steal(ret, p);
     281             : 
     282         293 :         ret[0] = p;
     283       57935 :         for (s = p, i=1; s < p+size; s++) {
     284       57642 :                 if (s[0] == '\n') {
     285        2428 :                         s[0] = 0;
     286        2428 :                         ret[i] = s+1;
     287        2428 :                         i++;
     288             :                 }
     289       57642 :                 if (s[0] == '\r') s[0] = 0;
     290             :         }
     291             : 
     292             :         /* remove any blank lines at the end */
     293         782 :         while (i > 0 && ret[i-1][0] == 0) {
     294         293 :                 i--;
     295             :         }
     296             : 
     297         293 :         if (numlines) *numlines = i;
     298             : 
     299         293 :         return ret;
     300             : }
     301             : 
     302             : 
     303             : /**
     304             : load a file into memory and return an array of pointers to lines in the file
     305             : must be freed with talloc_free().
     306             : **/
     307         111 : _PUBLIC_ char **file_lines_load(const char *fname, int *numlines, size_t maxsize, TALLOC_CTX *mem_ctx)
     308             : {
     309             :         char *p;
     310             :         size_t size;
     311             : 
     312         111 :         p = file_load(fname, &size, maxsize, mem_ctx);
     313         111 :         if (!p) return NULL;
     314             : 
     315          19 :         return file_lines_parse_internal(p, size, numlines, mem_ctx);
     316             : }
     317             : 
     318             : /**
     319             : load a fd into memory and return an array of pointers to lines in the file
     320             : must be freed with talloc_free(). If convert is true calls unix_to_dos on
     321             : the list.
     322             : **/
     323           0 : _PUBLIC_ char **fd_lines_load(int fd, int *numlines, size_t maxsize, TALLOC_CTX *mem_ctx)
     324             : {
     325             :         char *p;
     326             :         size_t size;
     327             : 
     328           0 :         p = fd_load(fd, &size, maxsize, mem_ctx);
     329           0 :         if (!p) return NULL;
     330             : 
     331           0 :         return file_lines_parse_internal(p, size, numlines, mem_ctx);
     332             : }
     333             : 
     334         274 : _PUBLIC_ char **file_lines_parse(const char *p_in,
     335             :                         size_t size,
     336             :                         int *numlines,
     337             :                         TALLOC_CTX *mem_ctx)
     338             : {
     339             :         /*
     340             :          * Copy the incoming string so it can end up
     341             :          * being owned by the returned pointer and
     342             :          * freed when that is.
     343             :          */
     344         274 :         char *p = talloc_strdup(mem_ctx, p_in);
     345         274 :         if (p == NULL) {
     346           0 :                 return NULL;
     347             :         }
     348         274 :         return file_lines_parse_internal(p, size, numlines, mem_ctx);
     349             : }
     350             : 
     351       16428 : _PUBLIC_ bool file_save_mode(const char *fname, const void *packet,
     352             :                              size_t length, mode_t mode)
     353             : {
     354             :         ssize_t num_written;
     355             :         int fd;
     356       16428 :         fd = open(fname, O_WRONLY|O_CREAT|O_EXCL, mode);
     357       16428 :         if (fd == -1) {
     358        7941 :                 return false;
     359             :         }
     360        8487 :         num_written = write(fd, packet, length);
     361        8487 :         if (num_written == -1 || (size_t)num_written != length) {
     362           0 :                 close(fd);
     363           0 :                 return false;
     364             :         }
     365        8487 :         close(fd);
     366        8487 :         return true;
     367             : }
     368             : 
     369             : /**
     370             :   save a lump of data into a file. Mostly used for debugging
     371             : */
     372       16403 : _PUBLIC_ bool file_save(const char *fname, const void *packet, size_t length)
     373             : {
     374       16403 :         return file_save_mode(fname, packet, length, 0644);
     375             : }
     376             : 
     377           0 : _PUBLIC_ int vfdprintf(int fd, const char *format, va_list ap)
     378             : {
     379             :         char *p;
     380             :         int len, ret;
     381             :         va_list ap2;
     382             : 
     383           0 :         va_copy(ap2, ap);
     384           0 :         len = vasprintf(&p, format, ap2);
     385           0 :         va_end(ap2);
     386           0 :         if (len <= 0) return len;
     387           0 :         ret = write(fd, p, len);
     388           0 :         SAFE_FREE(p);
     389           0 :         return ret;
     390             : }
     391             : 
     392           0 : _PUBLIC_ int fdprintf(int fd, const char *format, ...)
     393             : {
     394             :         va_list ap;
     395             :         int ret;
     396             : 
     397           0 :         va_start(ap, format);
     398           0 :         ret = vfdprintf(fd, format, ap);
     399           0 :         va_end(ap);
     400           0 :         return ret;
     401             : }
     402             : 
     403             : 
     404             : /*
     405             :   compare two files, return true if the two files have the same content
     406             :  */
     407           0 : bool file_compare(const char *path1, const char *path2)
     408             : {
     409           0 :         FILE *f1 = NULL, *f2 = NULL;
     410             :         uint8_t buf1[1024], buf2[1024];
     411           0 :         bool ret = false;
     412             : 
     413           0 :         f1 = fopen(path1, "r");
     414           0 :         if (f1 == NULL) {
     415           0 :                 goto done;
     416             :         }
     417           0 :         f2 = fopen(path2, "r");
     418           0 :         if (f2 == NULL) {
     419           0 :                 goto done;
     420             :         }
     421             : 
     422           0 :         while (!feof(f1)) {
     423           0 :                 size_t n1 = fread(buf1, 1, sizeof(buf1), f1);
     424           0 :                 size_t n2 = fread(buf2, 1, sizeof(buf2), f2);
     425             : 
     426           0 :                 if (n1 != n2) {
     427           0 :                         goto done;
     428             :                 }
     429           0 :                 if (n1 == 0) {
     430           0 :                         ret = (feof(f1) && feof(f2));
     431           0 :                         goto done;
     432             :                 }
     433           0 :                 if (memcmp(buf1, buf2, n1) != 0) {
     434           0 :                         goto done;
     435             :                 }
     436           0 :                 if (n1 < sizeof(buf1)) {
     437           0 :                         bool has_error = (ferror(f1) || ferror(f2));
     438           0 :                         if (has_error) {
     439           0 :                                 goto done;
     440             :                         }
     441             :                 }
     442             :         }
     443           0 :         ret = true;
     444           0 : done:
     445           0 :         if (f2 != NULL) {
     446           0 :                 fclose(f2);
     447             :         }
     448           0 :         if (f1 != NULL) {
     449           0 :                 fclose(f1);
     450             :         }
     451           0 :         return ret;
     452             : }
     453             : 
     454             : /**
     455             :  Load from a pipe into memory.
     456             : **/
     457           2 : char *file_ploadv(char * const argl[], size_t *size)
     458             : {
     459             :         int fd, n;
     460           2 :         char *p = NULL;
     461             :         char buf[1024];
     462             :         size_t total;
     463             : 
     464           2 :         fd = sys_popenv(argl);
     465           2 :         if (fd == -1) {
     466           0 :                 return NULL;
     467             :         }
     468             : 
     469           2 :         total = 0;
     470             : 
     471           6 :         while ((n = sys_read(fd, buf, sizeof(buf))) > 0) {
     472           2 :                 p = talloc_realloc(NULL, p, char, total + n + 1);
     473           2 :                 if (p == NULL) {
     474           0 :                         DBG_ERR("failed to expand buffer!\n");
     475           0 :                         close(fd);
     476           0 :                         return NULL;
     477             :                 }
     478           2 :                 memcpy(p+total, buf, n);
     479           2 :                 total += n;
     480             :         }
     481             : 
     482           2 :         if (p != NULL) {
     483           2 :                 p[total] = 0;
     484             :         }
     485             : 
     486             :         /*
     487             :          * FIXME: Perhaps ought to check that the command completed
     488             :          * successfully (returned 0); if not the data may be
     489             :          * truncated.
     490             :          */
     491           2 :         sys_pclose(fd);
     492             : 
     493           2 :         if (size) {
     494           2 :                 *size = total;
     495             :         }
     496             : 
     497           2 :         return p;
     498             : }

Generated by: LCOV version 1.13