Set umask security before calling mkstemp().
[cubemap] / util.cpp
1 #include <errno.h>
2 #include <stddef.h>
3 #include <stdio.h>
4 #include <stdlib.h>
5 #include <unistd.h>
6 #include <sys/stat.h>
7 #include <sys/types.h>
8
9 #include "log.h"
10 #include "util.h"
11
12 using namespace std;
13
14 int make_tempfile(const std::string &contents)
15 {
16         char filename[] = "/tmp/cubemap.XXXXXX";
17         mode_t old_umask = umask(0600);
18         int fd = mkstemp(filename);
19         umask(old_umask);
20         if (fd == -1) {
21                 log_perror("mkstemp");
22                 return -1;
23         }
24
25         if (unlink(filename) == -1) {
26                 log_perror("unlink");
27                 // Can still continue;
28         }
29
30         const char *ptr = contents.data();
31         size_t to_write = contents.size();
32         while (to_write > 0) {
33                 ssize_t ret = write(fd, ptr, to_write);
34                 if (ret == -1) {
35                         log_perror("write");
36                         close(fd);
37                         return -1;
38                 }
39
40                 ptr += ret;
41                 to_write -= ret;
42         }
43
44         return fd;
45 }
46
47 bool read_tempfile_and_close(int fd, std::string *contents)
48 {
49         bool ok = read_tempfile(fd, contents);
50
51         int ret;
52         do {
53                 ret = close(fd);  // Implicitly deletes the file.
54         } while (ret == -1 && errno == EINTR);
55         
56         if (ret == -1) {
57                 log_perror("close");
58                 // Can still continue.
59         }
60
61         return ok;
62 }
63
64 bool read_tempfile(int fd, std::string *contents)
65 {
66         ssize_t ret, has_read;
67
68         off_t len = lseek(fd, 0, SEEK_END);
69         if (len == -1) {
70                 log_perror("lseek");
71                 return false;
72         }
73
74         contents->resize(len);
75
76         if (lseek(fd, 0, SEEK_SET) == -1) {
77                 log_perror("lseek");
78                 return false;
79         }
80
81         has_read = 0;
82         while (has_read < len) {
83                 ret = read(fd, &((*contents)[has_read]), len - has_read);
84                 if (ret == -1) {
85                         log_perror("read");
86                         return false;
87                 }
88                 if (ret == 0) {
89                         log(ERROR, "Unexpected EOF!");
90                         return false;
91                 }
92                 has_read += ret;
93         }
94
95         return true;
96 }