2 This file is part of GNUnet.
3 Copyright (C) 2001--2013, 2016, 2018 GNUnet e.V.
5 GNUnet is free software: you can redistribute it and/or modify it
6 under the terms of the GNU Affero General Public License as published
7 by the Free Software Foundation, either version 3 of the License,
8 or (at your option) any later version.
10 GNUnet is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Affero General Public License for more details.
15 You should have received a copy of the GNU Affero General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>.
18 SPDX-License-Identifier: AGPL3.0-or-later
22 * @brief disk IO convenience methods
23 * @author Christian Grothoff
28 #include "gnunet_strings_lib.h"
29 #include "gnunet_disk_lib.h"
31 #define LOG(kind, ...) GNUNET_log_from (kind, "util-disk", __VA_ARGS__)
33 #define LOG_STRERROR(kind, syscall) \
34 GNUNET_log_from_strerror (kind, "util-disk", syscall)
36 #define LOG_STRERROR_FILE(kind, syscall, filename) \
37 GNUNET_log_from_strerror_file (kind, "util-disk", syscall, filename)
40 * Block size for IO for copying files.
42 #define COPY_BLK_SIZE 65536
44 #include <sys/types.h>
49 #include <sys/param.h>
52 #include <sys/mount.h>
54 #if HAVE_SYS_STATVFS_H
55 #include <sys/statvfs.h>
59 #define _IFMT 0170000 /* type of file */
60 #define _IFLNK 0120000 /* symbolic link */
61 #define S_ISLNK(m) (((m) & _IFMT) == _IFLNK)
66 * Handle used to manage a pipe.
68 struct GNUNET_DISK_PipeHandle
71 * File descriptors for the pipe.
72 * One or both of them could be NULL.
74 struct GNUNET_DISK_FileHandle *fd[2];
79 * Closure for the recursion to determine the file size
82 struct GetFileSizeData
85 * Set to the total file size.
90 * GNUNET_YES if symbolic links should be included.
92 int include_sym_links;
95 * GNUNET_YES if mode is file-only (return total == -1 for directories).
102 * Translate GNUnet-internal permission bitmap to UNIX file
103 * access permission bitmap.
105 * @param perm file permissions, GNUnet style
106 * @return file permissions, UNIX style
109 translate_unix_perms (enum GNUNET_DISK_AccessPermissions perm)
114 if (perm & GNUNET_DISK_PERM_USER_READ)
116 if (perm & GNUNET_DISK_PERM_USER_WRITE)
118 if (perm & GNUNET_DISK_PERM_USER_EXEC)
120 if (perm & GNUNET_DISK_PERM_GROUP_READ)
122 if (perm & GNUNET_DISK_PERM_GROUP_WRITE)
124 if (perm & GNUNET_DISK_PERM_GROUP_EXEC)
126 if (perm & GNUNET_DISK_PERM_OTHER_READ)
128 if (perm & GNUNET_DISK_PERM_OTHER_WRITE)
130 if (perm & GNUNET_DISK_PERM_OTHER_EXEC)
138 * Iterate over all files in the given directory and
139 * accumulate their size.
141 * @param cls closure of type `struct GetFileSizeData`
142 * @param fn current filename we are looking at
143 * @return #GNUNET_SYSERR on serious errors, otherwise #GNUNET_OK
146 getSizeRec (void *cls, const char *fn)
148 struct GetFileSizeData *gfsd = cls;
150 #if defined(HAVE_STAT64) && \
151 ! (defined(_FILE_OFFSET_BITS) && _FILE_OFFSET_BITS == 64)
154 if (0 != stat64 (fn, &buf))
156 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_DEBUG, "stat64", fn);
157 return GNUNET_SYSERR;
162 if (0 != stat (fn, &buf))
164 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_DEBUG, "stat", fn);
165 return GNUNET_SYSERR;
168 if ((S_ISDIR (buf.st_mode)) && (gfsd->single_file_mode == GNUNET_YES))
171 return GNUNET_SYSERR;
173 if ((! S_ISLNK (buf.st_mode)) || (gfsd->include_sym_links == GNUNET_YES))
174 gfsd->total += buf.st_size;
175 if ((S_ISDIR (buf.st_mode)) && (0 == access (fn, X_OK)) &&
176 ((! S_ISLNK (buf.st_mode)) || (gfsd->include_sym_links == GNUNET_YES)))
178 if (GNUNET_SYSERR == GNUNET_DISK_directory_scan (fn, &getSizeRec, gfsd))
179 return GNUNET_SYSERR;
186 * Checks whether a handle is invalid
188 * @param h handle to check
189 * @return #GNUNET_YES if invalid, #GNUNET_NO if valid
192 GNUNET_DISK_handle_invalid (const struct GNUNET_DISK_FileHandle *h)
194 return ((! h) || (h->fd == -1)) ? GNUNET_YES : GNUNET_NO;
199 * Get the size of an open file.
201 * @param fh open file handle
202 * @param size where to write size of the file
203 * @return #GNUNET_OK on success, #GNUNET_SYSERR on error
206 GNUNET_DISK_file_handle_size (struct GNUNET_DISK_FileHandle *fh, off_t *size)
210 if (0 != fstat (fh->fd, &sbuf))
211 return GNUNET_SYSERR;
212 *size = sbuf.st_size;
218 * Move the read/write pointer in a file
220 * @param h handle of an open file
221 * @param offset position to move to
222 * @param whence specification to which position the offset parameter relates to
223 * @return the new position on success, #GNUNET_SYSERR otherwise
226 GNUNET_DISK_file_seek (const struct GNUNET_DISK_FileHandle *h,
228 enum GNUNET_DISK_Seek whence)
233 return GNUNET_SYSERR;
236 static int t[] = { SEEK_SET, SEEK_CUR, SEEK_END };
238 return lseek (h->fd, offset, t[whence]);
243 * Get the size of the file (or directory) of the given file (in
246 * @param filename name of the file or directory
247 * @param size set to the size of the file (or,
248 * in the case of directories, the sum
249 * of all sizes of files in the directory)
250 * @param include_symbolic_links should symbolic links be
252 * @param single_file_mode #GNUNET_YES to only get size of one file
253 * and return #GNUNET_SYSERR for directories.
254 * @return #GNUNET_SYSERR on error, #GNUNET_OK on success
257 GNUNET_DISK_file_size (const char *filename,
259 int include_symbolic_links,
260 int single_file_mode)
262 struct GetFileSizeData gfsd;
265 GNUNET_assert (size != NULL);
267 gfsd.include_sym_links = include_symbolic_links;
268 gfsd.single_file_mode = single_file_mode;
269 ret = getSizeRec (&gfsd, filename);
276 * Obtain some unique identifiers for the given file
277 * that can be used to identify it in the local system.
278 * This function is used between GNUnet processes to
279 * quickly check if two files with the same absolute path
280 * are actually identical. The two processes represent
281 * the same peer but may communicate over the network
282 * (and the file may be on an NFS volume). This function
283 * may not be supported on all operating systems.
285 * @param filename name of the file
286 * @param dev set to the device ID
287 * @param ino set to the inode ID
288 * @return #GNUNET_OK on success
291 GNUNET_DISK_file_get_identifiers (const char *filename,
299 if (0 != stat (filename, &sbuf))
301 return GNUNET_SYSERR;
303 *ino = (uint64_t) sbuf.st_ino;
312 if (0 != statvfs (filename, &fbuf))
314 return GNUNET_SYSERR;
316 *dev = (uint64_t) fbuf.f_fsid;
322 if (0 != statfs (filename, &fbuf))
324 return GNUNET_SYSERR;
327 ((uint64_t) fbuf.f_fsid.val[0]) << 32 || ((uint64_t) fbuf.f_fsid.val[1]);
337 * Create the name for a temporary file or directory from a template.
339 * @param t template (without XXXXX or "/tmp/")
340 * @return name ready for passing to 'mktemp' or 'mkdtemp', NULL on error
343 mktemp_name (const char *t)
349 if ((t[0] != '/') && (t[0] != '\\'))
351 /* FIXME: This uses system codepage on W32, not UTF-8 */
352 tmpdir = getenv ("TMPDIR");
354 tmpdir = getenv ("TMP");
356 tmpdir = getenv ("TEMP");
359 GNUNET_asprintf (&tmpl, "%s/%s%s", tmpdir, t, "XXXXXX");
363 GNUNET_asprintf (&tmpl, "%s%s", t, "XXXXXX");
371 * Update POSIX permissions mask of a file on disk. If both argumets
372 * are #GNUNET_NO, the file is made world-read-write-executable (777).
374 * @param fn name of the file to update
375 * @param require_uid_match #GNUNET_YES means 700
376 * @param require_gid_match #GNUNET_YES means 770 unless @a require_uid_match is set
379 GNUNET_DISK_fix_permissions (const char *fn,
380 int require_uid_match,
381 int require_gid_match)
385 if (GNUNET_YES == require_uid_match)
386 mode = S_IRUSR | S_IWUSR | S_IXUSR;
387 else if (GNUNET_YES == require_gid_match)
388 mode = S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IWGRP | S_IXGRP;
390 mode = S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IWGRP | S_IXGRP | S_IROTH
392 if (0 != chmod (fn, mode))
393 GNUNET_log_strerror_file (GNUNET_ERROR_TYPE_WARNING, "chmod", fn);
398 * Create an (empty) temporary directory on disk. If the given name is not
399 * an absolute path, the current 'TMPDIR' will be prepended. In any case,
400 * 6 random characters will be appended to the name to create a unique
403 * @param t component to use for the name;
404 * does NOT contain "XXXXXX" or "/tmp/".
405 * @return NULL on error, otherwise name of fresh
406 * file on disk in directory for temporary files
409 GNUNET_DISK_mkdtemp (const char *t)
414 omask = umask (S_IWGRP | S_IWOTH | S_IRGRP | S_IROTH);
415 fn = mktemp_name (t);
416 if (fn != mkdtemp (fn))
418 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_ERROR, "mkdtemp", fn);
429 * Move a file out of the way (create a backup) by
430 * renaming it to "orig.NUM~" where NUM is the smallest
431 * number that is not used yet.
433 * @param fil name of the file to back up
436 GNUNET_DISK_file_backup (const char *fil)
442 slen = strlen (fil) + 20;
443 target = GNUNET_malloc (slen);
447 GNUNET_snprintf (target, slen, "%s.%u~", fil, num++);
449 while (0 == access (target, F_OK));
450 if (0 != rename (fil, target))
451 GNUNET_log_strerror_file (GNUNET_ERROR_TYPE_ERROR, "rename", fil);
452 GNUNET_free (target);
457 * Create an (empty) temporary file on disk. If the given name is not
458 * an absolute path, the current 'TMPDIR' will be prepended. In any case,
459 * 6 random characters will be appended to the name to create a unique
462 * @param t component to use for the name;
463 * does NOT contain "XXXXXX" or "/tmp/".
464 * @return NULL on error, otherwise name of fresh
465 * file on disk in directory for temporary files
468 GNUNET_DISK_mktemp (const char *t)
474 omask = umask (S_IWGRP | S_IWOTH | S_IRGRP | S_IROTH);
475 fn = mktemp_name (t);
476 if (-1 == (fd = mkstemp (fn)))
478 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_ERROR, "mkstemp", fn);
485 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "close", fn);
491 * Test if @a fil is a directory and listable. Optionally, also check if the
492 * directory is readable. Will not print an error message if the directory does
493 * not exist. Will log errors if #GNUNET_SYSERR is returned (i.e., a file exists
494 * with the same name).
496 * @param fil filename to test
497 * @param is_readable #GNUNET_YES to additionally check if @a fil is readable;
498 * #GNUNET_NO to disable this check
499 * @return #GNUNET_YES if yes, #GNUNET_NO if not; #GNUNET_SYSERR if it
500 * does not exist or stat'ed
503 GNUNET_DISK_directory_test (const char *fil, int is_readable)
505 struct stat filestat;
508 ret = stat (fil, &filestat);
512 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "stat", fil);
513 return GNUNET_SYSERR;
515 if (! S_ISDIR (filestat.st_mode))
517 LOG (GNUNET_ERROR_TYPE_INFO,
518 "A file already exits with the same name %s\n",
522 if (GNUNET_YES == is_readable)
523 ret = access (fil, R_OK | X_OK);
525 ret = access (fil, X_OK);
528 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "access", fil);
536 * Check that fil corresponds to a filename
537 * (of a file that exists and that is not a directory).
539 * @param fil filename to check
540 * @return #GNUNET_YES if yes, #GNUNET_NO if not a file, #GNUNET_SYSERR if something
541 * else (will print an error message in that case, too).
544 GNUNET_DISK_file_test (const char *fil)
546 struct stat filestat;
550 rdir = GNUNET_STRINGS_filename_expand (fil);
552 return GNUNET_SYSERR;
554 ret = stat (rdir, &filestat);
559 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "stat", rdir);
561 return GNUNET_SYSERR;
566 if (! S_ISREG (filestat.st_mode))
571 if (access (rdir, F_OK) < 0)
573 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "access", rdir);
575 return GNUNET_SYSERR;
583 * Implementation of "mkdir -p"
585 * @param dir the directory to create
586 * @return #GNUNET_OK on success, #GNUNET_SYSERR on failure
589 GNUNET_DISK_directory_create (const char *dir)
597 rdir = GNUNET_STRINGS_filename_expand (dir);
601 return GNUNET_SYSERR;
606 pos = 1; /* skip heading '/' */
608 /* Check which low level directories already exist */
610 rdir[len] = DIR_SEPARATOR;
613 if (DIR_SEPARATOR == rdir[pos2])
616 ret = GNUNET_DISK_directory_test (rdir, GNUNET_NO);
617 if (GNUNET_NO == ret)
619 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
620 "Creating directory `%s' failed",
623 return GNUNET_SYSERR;
625 rdir[pos2] = DIR_SEPARATOR;
626 if (GNUNET_YES == ret)
637 /* Start creating directories */
640 if ((rdir[pos] == DIR_SEPARATOR) || (pos == len))
643 ret = GNUNET_DISK_directory_test (rdir, GNUNET_NO);
644 if (GNUNET_NO == ret)
646 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
647 "Creating directory `%s' failed",
650 return GNUNET_SYSERR;
652 if (GNUNET_SYSERR == ret)
655 S_IRUSR | S_IWUSR | S_IXUSR | S_IRGRP | S_IXGRP | S_IROTH
656 | S_IXOTH); /* 755 */
658 if ((ret != 0) && (errno != EEXIST))
660 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_ERROR, "mkdir", rdir);
662 return GNUNET_SYSERR;
665 rdir[pos] = DIR_SEPARATOR;
675 * Create the directory structure for storing a file.
677 * @param filename name of a file in the directory
678 * @returns #GNUNET_OK on success,
679 * #GNUNET_SYSERR on failure,
680 * #GNUNET_NO if the directory
681 * exists but is not writeable for us
684 GNUNET_DISK_directory_create_for_file (const char *filename)
691 rdir = GNUNET_STRINGS_filename_expand (filename);
695 return GNUNET_SYSERR;
697 if (0 == access (rdir, W_OK))
704 while ((len > 0) && (rdir[len] != DIR_SEPARATOR))
707 /* The empty path is invalid and in this case refers to / */
711 rdir = GNUNET_strdup ("/");
713 ret = GNUNET_DISK_directory_create (rdir);
714 if ((GNUNET_OK == ret) && (0 != access (rdir, W_OK)))
724 * Read the contents of a binary file into a buffer.
726 * @param h handle to an open file
727 * @param result the buffer to write the result to
728 * @param len the maximum number of bytes to read
729 * @return the number of bytes read on success, #GNUNET_SYSERR on failure
732 GNUNET_DISK_file_read (const struct GNUNET_DISK_FileHandle *h,
739 return GNUNET_SYSERR;
742 return read (h->fd, result, len);
747 * Read the contents of a binary file into a buffer.
748 * Guarantees not to block (returns GNUNET_SYSERR and sets errno to EAGAIN
749 * when no data can be read).
751 * @param h handle to an open file
752 * @param result the buffer to write the result to
753 * @param len the maximum number of bytes to read
754 * @return the number of bytes read on success, #GNUNET_SYSERR on failure
757 GNUNET_DISK_file_read_non_blocking (const struct GNUNET_DISK_FileHandle *h,
764 return GNUNET_SYSERR;
770 /* set to non-blocking, read, then set back */
771 flags = fcntl (h->fd, F_GETFL);
772 if (0 == (flags & O_NONBLOCK))
773 (void) fcntl (h->fd, F_SETFL, flags | O_NONBLOCK);
774 ret = read (h->fd, result, len);
775 if (0 == (flags & O_NONBLOCK))
778 (void) fcntl (h->fd, F_SETFL, flags);
786 * Read the contents of a binary file into a buffer.
788 * @param fn file name
789 * @param result the buffer to write the result to
790 * @param len the maximum number of bytes to read
791 * @return number of bytes read, #GNUNET_SYSERR on failure
794 GNUNET_DISK_fn_read (const char *fn, void *result, size_t len)
796 struct GNUNET_DISK_FileHandle *fh;
800 fh = GNUNET_DISK_file_open (fn, GNUNET_DISK_OPEN_READ, GNUNET_DISK_PERM_NONE);
802 return GNUNET_SYSERR;
803 ret = GNUNET_DISK_file_read (fh, result, len);
805 GNUNET_assert (GNUNET_OK == GNUNET_DISK_file_close (fh));
812 * Write a buffer to a file.
814 * @param h handle to open file
815 * @param buffer the data to write
816 * @param n number of bytes to write
817 * @return number of bytes written on success, #GNUNET_SYSERR on error
820 GNUNET_DISK_file_write (const struct GNUNET_DISK_FileHandle *h,
827 return GNUNET_SYSERR;
830 return write (h->fd, buffer, n);
835 * Write a buffer to a file, blocking, if necessary.
837 * @param h handle to open file
838 * @param buffer the data to write
839 * @param n number of bytes to write
840 * @return number of bytes written on success, #GNUNET_SYSERR on error
843 GNUNET_DISK_file_write_blocking (const struct GNUNET_DISK_FileHandle *h,
850 return GNUNET_SYSERR;
857 /* set to blocking, write, then set back */
858 flags = fcntl (h->fd, F_GETFL);
859 if (0 != (flags & O_NONBLOCK))
860 (void) fcntl (h->fd, F_SETFL, flags - O_NONBLOCK);
861 ret = write (h->fd, buffer, n);
862 if (0 == (flags & O_NONBLOCK))
863 (void) fcntl (h->fd, F_SETFL, flags);
869 * Write a buffer to a file. If the file is longer than the
870 * number of bytes that will be written, it will be truncated.
872 * @param fn file name
873 * @param buffer the data to write
874 * @param n number of bytes to write
875 * @param mode file permissions
876 * @return number of bytes written on success, #GNUNET_SYSERR on error
879 GNUNET_DISK_fn_write (const char *fn,
882 enum GNUNET_DISK_AccessPermissions mode)
884 struct GNUNET_DISK_FileHandle *fh;
888 GNUNET_DISK_file_open (fn,
889 GNUNET_DISK_OPEN_WRITE | GNUNET_DISK_OPEN_TRUNCATE
890 | GNUNET_DISK_OPEN_CREATE,
893 return GNUNET_SYSERR;
894 ret = GNUNET_DISK_file_write (fh, buffer, n);
895 GNUNET_assert (GNUNET_OK == GNUNET_DISK_file_close (fh));
901 * Scan a directory for files.
903 * @param dir_name the name of the directory
904 * @param callback the method to call for each file,
905 * can be NULL, in that case, we only count
906 * @param callback_cls closure for @a callback
907 * @return the number of files found, #GNUNET_SYSERR on error or
908 * ieration aborted by callback returning #GNUNET_SYSERR
911 GNUNET_DISK_directory_scan (const char *dir_name,
912 GNUNET_FileNameCallback callback,
916 struct dirent *finfo;
922 unsigned int name_len;
925 GNUNET_assert (NULL != dir_name);
926 dname = GNUNET_STRINGS_filename_expand (dir_name);
928 return GNUNET_SYSERR;
929 while ((strlen (dname) > 0) && (dname[strlen (dname) - 1] == DIR_SEPARATOR))
930 dname[strlen (dname) - 1] = '\0';
931 if (0 != stat (dname, &istat))
933 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "stat", dname);
935 return GNUNET_SYSERR;
937 if (! S_ISDIR (istat.st_mode))
939 LOG (GNUNET_ERROR_TYPE_WARNING,
940 _ ("Expected `%s' to be a directory!\n"),
943 return GNUNET_SYSERR;
946 dinfo = opendir (dname);
947 if ((EACCES == errno) || (NULL == dinfo))
949 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "opendir", dname);
953 return GNUNET_SYSERR;
956 n_size = strlen (dname) + name_len + strlen (DIR_SEPARATOR_STR) + 1;
957 name = GNUNET_malloc (n_size);
958 while (NULL != (finfo = readdir (dinfo)))
960 if ((0 == strcmp (finfo->d_name, ".")) ||
961 (0 == strcmp (finfo->d_name, "..")))
963 if (NULL != callback)
965 if (name_len < strlen (finfo->d_name))
968 name_len = strlen (finfo->d_name);
969 n_size = strlen (dname) + name_len + strlen (DIR_SEPARATOR_STR) + 1;
970 name = GNUNET_malloc (n_size);
972 /* dname can end in "/" only if dname == "/";
973 * if dname does not end in "/", we need to add
974 * a "/" (otherwise, we must not!) */
975 GNUNET_snprintf (name,
979 (0 == strcmp (dname, DIR_SEPARATOR_STR))
983 ret = callback (callback_cls, name);
984 if (GNUNET_OK != ret)
989 if (GNUNET_NO == ret)
991 return GNUNET_SYSERR;
1004 * Function that removes the given directory by calling
1005 * #GNUNET_DISK_directory_remove().
1007 * @param unused not used
1008 * @param fn directory to remove
1009 * @return #GNUNET_OK
1012 remove_helper (void *unused, const char *fn)
1015 (void) GNUNET_DISK_directory_remove (fn);
1021 * Remove all files in a directory (rm -r). Call with
1024 * @param filename the file to remove
1025 * @return #GNUNET_OK on success, #GNUNET_SYSERR on error
1028 GNUNET_DISK_directory_remove (const char *filename)
1032 if (NULL == filename)
1035 return GNUNET_SYSERR;
1037 if (0 != lstat (filename, &istat))
1038 return GNUNET_NO; /* file may not exist... */
1039 (void) chmod (filename, S_IWUSR | S_IRUSR | S_IXUSR);
1040 if (0 == unlink (filename))
1042 if ((errno != EISDIR) &&
1043 /* EISDIR is not sufficient in all cases, e.g.
1044 * sticky /tmp directory may result in EPERM on BSD.
1045 * So we also explicitly check "isDirectory" */
1046 (GNUNET_YES != GNUNET_DISK_directory_test (filename, GNUNET_YES)))
1048 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "rmdir", filename);
1049 return GNUNET_SYSERR;
1051 if (GNUNET_SYSERR ==
1052 GNUNET_DISK_directory_scan (filename, &remove_helper, NULL))
1053 return GNUNET_SYSERR;
1054 if (0 != rmdir (filename))
1056 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "rmdir", filename);
1057 return GNUNET_SYSERR;
1066 * @param src file to copy
1067 * @param dst destination file name
1068 * @return #GNUNET_OK on success, #GNUNET_SYSERR on error
1071 GNUNET_DISK_file_copy (const char *src, const char *dst)
1078 struct GNUNET_DISK_FileHandle *in;
1079 struct GNUNET_DISK_FileHandle *out;
1081 if (GNUNET_OK != GNUNET_DISK_file_size (src, &size, GNUNET_YES, GNUNET_YES))
1083 GNUNET_log_strerror_file (GNUNET_ERROR_TYPE_ERROR, "stat", src);
1084 return GNUNET_SYSERR;
1088 GNUNET_DISK_file_open (src, GNUNET_DISK_OPEN_READ, GNUNET_DISK_PERM_NONE);
1091 GNUNET_log_strerror_file (GNUNET_ERROR_TYPE_ERROR, "open", src);
1092 return GNUNET_SYSERR;
1095 GNUNET_DISK_file_open (dst,
1096 GNUNET_DISK_OPEN_WRITE | GNUNET_DISK_OPEN_CREATE
1097 | GNUNET_DISK_OPEN_FAILIFEXISTS,
1098 GNUNET_DISK_PERM_USER_READ
1099 | GNUNET_DISK_PERM_USER_WRITE
1100 | GNUNET_DISK_PERM_GROUP_READ
1101 | GNUNET_DISK_PERM_GROUP_WRITE);
1104 GNUNET_log_strerror_file (GNUNET_ERROR_TYPE_ERROR, "open", dst);
1105 GNUNET_DISK_file_close (in);
1106 return GNUNET_SYSERR;
1108 buf = GNUNET_malloc (COPY_BLK_SIZE);
1111 len = COPY_BLK_SIZE;
1112 if (len > size - pos)
1114 sret = GNUNET_DISK_file_read (in, buf, len);
1115 if ((sret < 0) || (len != (size_t) sret))
1117 sret = GNUNET_DISK_file_write (out, buf, len);
1118 if ((sret < 0) || (len != (size_t) sret))
1123 GNUNET_DISK_file_close (in);
1124 GNUNET_DISK_file_close (out);
1128 GNUNET_DISK_file_close (in);
1129 GNUNET_DISK_file_close (out);
1130 return GNUNET_SYSERR;
1135 * @brief Removes special characters as ':' from a filename.
1136 * @param fn the filename to canonicalize
1139 GNUNET_DISK_filename_canonicalize (char *fn)
1144 for (idx = fn; *idx; idx++)
1148 if ((c == '/') || (c == '\\') || (c == ':') || (c == '*') || (c == '?') ||
1152 (c == '<') || (c == '>') || (c == '|') )
1161 * @brief Change owner of a file
1163 * @param filename name of file to change the owner of
1164 * @param user name of the new owner
1165 * @return #GNUNET_OK on success, #GNUNET_SYSERR on failure
1168 GNUNET_DISK_file_change_owner (const char *filename, const char *user)
1172 pws = getpwnam (user);
1175 LOG (GNUNET_ERROR_TYPE_ERROR,
1176 _ ("Cannot obtain information about user `%s': %s\n"),
1179 return GNUNET_SYSERR;
1181 if (0 != chown (filename, pws->pw_uid, pws->pw_gid))
1183 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "chown", filename);
1184 return GNUNET_SYSERR;
1191 * Lock a part of a file
1193 * @param fh file handle
1194 * @param lock_start absolute position from where to lock
1195 * @param lock_end absolute position until where to lock
1196 * @param excl #GNUNET_YES for an exclusive lock
1197 * @return #GNUNET_OK on success, #GNUNET_SYSERR on error
1200 GNUNET_DISK_file_lock (struct GNUNET_DISK_FileHandle *fh,
1208 return GNUNET_SYSERR;
1213 memset (&fl, 0, sizeof(struct flock));
1214 fl.l_type = excl ? F_WRLCK : F_RDLCK;
1215 fl.l_whence = SEEK_SET;
1216 fl.l_start = lock_start;
1217 fl.l_len = lock_end;
1219 return fcntl (fh->fd, F_SETLK, &fl) != 0 ? GNUNET_SYSERR : GNUNET_OK;
1224 * Unlock a part of a file
1226 * @param fh file handle
1227 * @param unlock_start absolute position from where to unlock
1228 * @param unlock_end absolute position until where to unlock
1229 * @return #GNUNET_OK on success, #GNUNET_SYSERR on error
1232 GNUNET_DISK_file_unlock (struct GNUNET_DISK_FileHandle *fh,
1239 return GNUNET_SYSERR;
1244 memset (&fl, 0, sizeof(struct flock));
1245 fl.l_type = F_UNLCK;
1246 fl.l_whence = SEEK_SET;
1247 fl.l_start = unlock_start;
1248 fl.l_len = unlock_end;
1250 return fcntl (fh->fd, F_SETLK, &fl) != 0 ? GNUNET_SYSERR : GNUNET_OK;
1255 * Open a file. Note that the access permissions will only be
1256 * used if a new file is created and if the underlying operating
1257 * system supports the given permissions.
1259 * @param fn file name to be opened
1260 * @param flags opening flags, a combination of GNUNET_DISK_OPEN_xxx bit flags
1261 * @param perm permissions for the newly created file, use
1262 * #GNUNET_DISK_PERM_NONE if a file could not be created by this
1263 * call (because of flags)
1264 * @return IO handle on success, NULL on error
1266 struct GNUNET_DISK_FileHandle *
1267 GNUNET_DISK_file_open (const char *fn,
1268 enum GNUNET_DISK_OpenFlags flags,
1269 enum GNUNET_DISK_AccessPermissions perm)
1272 struct GNUNET_DISK_FileHandle *ret;
1278 expfn = GNUNET_STRINGS_filename_expand (fn);
1283 if (GNUNET_DISK_OPEN_READWRITE == (flags & GNUNET_DISK_OPEN_READWRITE))
1284 oflags = O_RDWR; /* note: O_RDWR is NOT always O_RDONLY | O_WRONLY */
1285 else if (flags & GNUNET_DISK_OPEN_READ)
1287 else if (flags & GNUNET_DISK_OPEN_WRITE)
1292 GNUNET_free (expfn);
1295 if (flags & GNUNET_DISK_OPEN_FAILIFEXISTS)
1296 oflags |= (O_CREAT | O_EXCL);
1297 if (flags & GNUNET_DISK_OPEN_TRUNCATE)
1299 if (flags & GNUNET_DISK_OPEN_APPEND)
1301 if (GNUNET_NO == GNUNET_DISK_file_test (fn))
1303 if (flags & GNUNET_DISK_OPEN_CREATE)
1305 (void) GNUNET_DISK_directory_create_for_file (expfn);
1307 mode = translate_unix_perms (perm);
1320 if (0 == (flags & GNUNET_DISK_OPEN_FAILIFEXISTS))
1321 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_WARNING, "open", expfn);
1323 LOG_STRERROR_FILE (GNUNET_ERROR_TYPE_DEBUG, "open", expfn);
1324 GNUNET_free (expfn);
1328 ret = GNUNET_new (struct GNUNET_DISK_FileHandle);
1332 GNUNET_free (expfn);
1338 * Close an open file.
1340 * @param h file handle
1341 * @return #GNUNET_OK on success, #GNUNET_SYSERR otherwise
1344 GNUNET_DISK_file_close (struct GNUNET_DISK_FileHandle *h)
1351 return GNUNET_SYSERR;
1356 if (close (h->fd) != 0)
1358 LOG_STRERROR (GNUNET_ERROR_TYPE_WARNING, "close");
1359 ret = GNUNET_SYSERR;
1368 * Get a handle from a native integer FD.
1370 * @param fno native integer file descriptor
1371 * @return file handle corresponding to the descriptor, NULL on error
1373 struct GNUNET_DISK_FileHandle *
1374 GNUNET_DISK_get_handle_from_int_fd (int fno)
1376 struct GNUNET_DISK_FileHandle *fh;
1378 if ((((off_t) -1) == lseek (fno, 0, SEEK_CUR)) && (EBADF == errno))
1379 return NULL; /* invalid FD */
1381 fh = GNUNET_new (struct GNUNET_DISK_FileHandle);
1390 * Get a handle from a native streaming FD.
1392 * @param fd native streaming file descriptor
1393 * @return file handle corresponding to the descriptor
1395 struct GNUNET_DISK_FileHandle *
1396 GNUNET_DISK_get_handle_from_native (FILE *fd)
1404 return GNUNET_DISK_get_handle_from_int_fd (fno);
1409 * Handle for a memory-mapping operation.
1411 struct GNUNET_DISK_MapHandle
1414 * Address where the map is in memory.
1419 * Number of bytes mapped.
1426 #define MAP_FAILED ((void *) -1)
1430 * Map a file into memory
1432 * @param h open file handle
1433 * @param m handle to the new mapping
1434 * @param access access specification, GNUNET_DISK_MAP_TYPE_xxx
1435 * @param len size of the mapping
1436 * @return pointer to the mapped memory region, NULL on failure
1439 GNUNET_DISK_file_map (const struct GNUNET_DISK_FileHandle *h,
1440 struct GNUNET_DISK_MapHandle **m,
1441 enum GNUNET_DISK_MapType access,
1453 if (access & GNUNET_DISK_MAP_TYPE_READ)
1455 if (access & GNUNET_DISK_MAP_TYPE_WRITE)
1457 *m = GNUNET_new (struct GNUNET_DISK_MapHandle);
1458 (*m)->addr = mmap (NULL, len, prot, MAP_SHARED, h->fd, 0);
1459 GNUNET_assert (NULL != (*m)->addr);
1460 if (MAP_FAILED == (*m)->addr)
1472 * @param h mapping handle
1473 * @return GNUNET_OK on success, GNUNET_SYSERR otherwise
1476 GNUNET_DISK_file_unmap (struct GNUNET_DISK_MapHandle *h)
1483 return GNUNET_SYSERR;
1486 ret = munmap (h->addr, h->len) != -1 ? GNUNET_OK : GNUNET_SYSERR;
1494 * Write file changes to disk
1495 * @param h handle to an open file
1496 * @return GNUNET_OK on success, GNUNET_SYSERR otherwise
1499 GNUNET_DISK_file_sync (const struct GNUNET_DISK_FileHandle *h)
1504 return GNUNET_SYSERR;
1507 #if ! defined(__linux__) || ! defined(GNU)
1508 return fsync (h->fd) == -1 ? GNUNET_SYSERR : GNUNET_OK;
1510 return fdatasync (h->fd) == -1 ? GNUNET_SYSERR : GNUNET_OK;
1516 * Creates an interprocess channel
1518 * @param blocking_read creates an asynchronous pipe for reading if set to GNUNET_NO
1519 * @param blocking_write creates an asynchronous pipe for writing if set to GNUNET_NO
1520 * @param inherit_read inherit the parent processes stdin (only for windows)
1521 * @param inherit_write inherit the parent processes stdout (only for windows)
1522 * @return handle to the new pipe, NULL on error
1524 struct GNUNET_DISK_PipeHandle *
1525 GNUNET_DISK_pipe (int blocking_read,
1534 (void) inherit_read;
1535 (void) inherit_write;
1540 LOG_STRERROR (GNUNET_ERROR_TYPE_ERROR, "pipe");
1544 return GNUNET_DISK_pipe_from_fd (blocking_read, blocking_write, fd);
1549 * Creates a pipe object from a couple of file descriptors.
1550 * Useful for wrapping existing pipe FDs.
1552 * @param blocking_read creates an asynchronous pipe for reading if set to GNUNET_NO
1553 * @param blocking_write creates an asynchronous pipe for writing if set to GNUNET_NO
1554 * @param fd an array of two fd values. One of them may be -1 for read-only or write-only pipes
1556 * @return handle to the new pipe, NULL on error
1558 struct GNUNET_DISK_PipeHandle *
1559 GNUNET_DISK_pipe_from_fd (int blocking_read, int blocking_write, int fd[2])
1561 struct GNUNET_DISK_PipeHandle *p;
1563 p = GNUNET_new (struct GNUNET_DISK_PipeHandle);
1567 int eno = 0; /* make gcc happy */
1572 p->fd[0] = GNUNET_new (struct GNUNET_DISK_FileHandle);
1573 p->fd[0]->fd = fd[0];
1574 if (! blocking_read)
1576 flags = fcntl (fd[0], F_GETFL);
1577 flags |= O_NONBLOCK;
1578 if (0 > fcntl (fd[0], F_SETFL, flags))
1584 flags = fcntl (fd[0], F_GETFD);
1585 flags |= FD_CLOEXEC;
1586 if (0 > fcntl (fd[0], F_SETFD, flags))
1595 p->fd[1] = GNUNET_new (struct GNUNET_DISK_FileHandle);
1596 p->fd[1]->fd = fd[1];
1597 if (! blocking_write)
1599 flags = fcntl (fd[1], F_GETFL);
1600 flags |= O_NONBLOCK;
1601 if (0 > fcntl (fd[1], F_SETFL, flags))
1607 flags = fcntl (fd[1], F_GETFD);
1608 flags |= FD_CLOEXEC;
1609 if (0 > fcntl (fd[1], F_SETFD, flags))
1618 LOG_STRERROR (GNUNET_ERROR_TYPE_ERROR, "fcntl");
1619 if (p->fd[0]->fd >= 0)
1620 GNUNET_break (0 == close (p->fd[0]->fd));
1621 if (p->fd[1]->fd >= 0)
1622 GNUNET_break (0 == close (p->fd[1]->fd));
1623 GNUNET_free_non_null (p->fd[0]);
1624 GNUNET_free_non_null (p->fd[1]);
1635 * Closes an interprocess channel
1637 * @param p pipe to close
1638 * @param end which end of the pipe to close
1639 * @return GNUNET_OK on success, GNUNET_SYSERR otherwise
1642 GNUNET_DISK_pipe_close_end (struct GNUNET_DISK_PipeHandle *p,
1643 enum GNUNET_DISK_PipeEnd end)
1645 int ret = GNUNET_OK;
1647 if (end == GNUNET_DISK_PIPE_END_READ)
1651 ret = GNUNET_DISK_file_close (p->fd[0]);
1655 else if (end == GNUNET_DISK_PIPE_END_WRITE)
1659 ret = GNUNET_DISK_file_close (p->fd[1]);
1669 * Detaches one of the ends from the pipe.
1670 * Detached end is a fully-functional FileHandle, it will
1671 * not be affected by anything you do with the pipe afterwards.
1672 * Each end of a pipe can only be detched from it once (i.e.
1673 * it is not duplicated).
1675 * @param p pipe to detach an end from
1676 * @param end which end of the pipe to detach
1677 * @return Detached end on success, NULL on failure
1678 * (or if that end is not present or is closed).
1680 struct GNUNET_DISK_FileHandle *
1681 GNUNET_DISK_pipe_detach_end (struct GNUNET_DISK_PipeHandle *p,
1682 enum GNUNET_DISK_PipeEnd end)
1684 struct GNUNET_DISK_FileHandle *ret = NULL;
1686 if (end == GNUNET_DISK_PIPE_END_READ)
1694 else if (end == GNUNET_DISK_PIPE_END_WRITE)
1708 * Closes an interprocess channel
1710 * @param p pipe to close
1711 * @return GNUNET_OK on success, GNUNET_SYSERR otherwise
1714 GNUNET_DISK_pipe_close (struct GNUNET_DISK_PipeHandle *p)
1716 int ret = GNUNET_OK;
1719 int write_end_close;
1720 int read_end_close_errno;
1721 int write_end_close_errno;
1723 read_end_close = GNUNET_DISK_pipe_close_end (p, GNUNET_DISK_PIPE_END_READ);
1724 read_end_close_errno = errno;
1725 write_end_close = GNUNET_DISK_pipe_close_end (p, GNUNET_DISK_PIPE_END_WRITE);
1726 write_end_close_errno = errno;
1729 if (GNUNET_OK != read_end_close)
1731 errno = read_end_close_errno;
1732 ret = read_end_close;
1734 else if (GNUNET_OK != write_end_close)
1736 errno = write_end_close_errno;
1737 ret = write_end_close;
1745 * Get the handle to a particular pipe end
1748 * @param n end to access
1749 * @return handle for the respective end
1751 const struct GNUNET_DISK_FileHandle *
1752 GNUNET_DISK_pipe_handle (const struct GNUNET_DISK_PipeHandle *p,
1753 enum GNUNET_DISK_PipeEnd n)
1757 case GNUNET_DISK_PIPE_END_READ:
1758 case GNUNET_DISK_PIPE_END_WRITE:
1769 * Retrieve OS file handle
1771 * @param fh GNUnet file descriptor
1772 * @param dst destination buffer
1773 * @param dst_len length of dst
1774 * @return #GNUNET_OK on success, #GNUNET_SYSERR otherwise
1777 GNUNET_DISK_internal_file_handle_ (const struct GNUNET_DISK_FileHandle *fh,
1782 return GNUNET_SYSERR;
1784 if (dst_len < sizeof(int))
1785 return GNUNET_SYSERR;
1786 *((int *) dst) = fh->fd;
1793 * Helper function for #GNUNET_DISK_purge_cfg_dir.
1795 * @param cls a `const char *` with the option to purge
1796 * @param cfg our configuration
1797 * @return #GNUNET_OK on success
1800 purge_cfg_dir (void *cls, const struct GNUNET_CONFIGURATION_Handle *cfg)
1802 const char *option = cls;
1806 GNUNET_CONFIGURATION_get_value_filename (cfg, "PATHS", option, &tmpname))
1808 GNUNET_log_config_missing (GNUNET_ERROR_TYPE_ERROR, "PATHS", option);
1811 if (GNUNET_SYSERR == GNUNET_DISK_directory_remove (tmpname))
1813 GNUNET_log_strerror_file (GNUNET_ERROR_TYPE_ERROR, "remove", tmpname);
1814 GNUNET_free (tmpname);
1817 GNUNET_free (tmpname);
1823 * Remove the directory given under @a option in
1824 * section [PATHS] in configuration under @a cfg_filename
1826 * @param cfg_filename configuration file to parse
1827 * @param option option with the dir name to purge
1830 GNUNET_DISK_purge_cfg_dir (const char *cfg_filename, const char *option)
1832 GNUNET_break (GNUNET_OK ==
1833 GNUNET_CONFIGURATION_parse_and_run (cfg_filename,