xref: /aosp_15_r20/external/skia/src/ports/SkOSFile_posix.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2013 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkString.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTFitsIn.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTemplates.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkOSFile.h"
13*c8dee2aaSAndroid Build Coastguard Worker 
14*c8dee2aaSAndroid Build Coastguard Worker #include <dirent.h>
15*c8dee2aaSAndroid Build Coastguard Worker #include <new>
16*c8dee2aaSAndroid Build Coastguard Worker #include <stdio.h>
17*c8dee2aaSAndroid Build Coastguard Worker #include <string.h>
18*c8dee2aaSAndroid Build Coastguard Worker #include <sys/mman.h>
19*c8dee2aaSAndroid Build Coastguard Worker #include <sys/stat.h>
20*c8dee2aaSAndroid Build Coastguard Worker #include <sys/types.h>
21*c8dee2aaSAndroid Build Coastguard Worker #include <unistd.h>
22*c8dee2aaSAndroid Build Coastguard Worker 
23*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_BUILD_FOR_IOS
24*c8dee2aaSAndroid Build Coastguard Worker #include "src/ports/SkOSFile_ios.h"
25*c8dee2aaSAndroid Build Coastguard Worker #endif
26*c8dee2aaSAndroid Build Coastguard Worker 
sk_fsync(FILE * f)27*c8dee2aaSAndroid Build Coastguard Worker void sk_fsync(FILE* f) {
28*c8dee2aaSAndroid Build Coastguard Worker #if !defined(SK_BUILD_FOR_ANDROID) && !defined(__UCLIBC__) && !defined(_NEWLIB_VERSION)
29*c8dee2aaSAndroid Build Coastguard Worker     int fd = fileno(f);
30*c8dee2aaSAndroid Build Coastguard Worker     fsync(fd);
31*c8dee2aaSAndroid Build Coastguard Worker #endif
32*c8dee2aaSAndroid Build Coastguard Worker }
33*c8dee2aaSAndroid Build Coastguard Worker 
sk_exists(const char * path,SkFILE_Flags flags)34*c8dee2aaSAndroid Build Coastguard Worker bool sk_exists(const char *path, SkFILE_Flags flags) {
35*c8dee2aaSAndroid Build Coastguard Worker     int mode = F_OK;
36*c8dee2aaSAndroid Build Coastguard Worker     if (flags & kRead_SkFILE_Flag) {
37*c8dee2aaSAndroid Build Coastguard Worker         mode |= R_OK;
38*c8dee2aaSAndroid Build Coastguard Worker     }
39*c8dee2aaSAndroid Build Coastguard Worker     if (flags & kWrite_SkFILE_Flag) {
40*c8dee2aaSAndroid Build Coastguard Worker         mode |= W_OK;
41*c8dee2aaSAndroid Build Coastguard Worker     }
42*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_BUILD_FOR_IOS
43*c8dee2aaSAndroid Build Coastguard Worker     // if the default path fails, check the bundle (but only if read-only)
44*c8dee2aaSAndroid Build Coastguard Worker     if (0 == access(path, mode)) {
45*c8dee2aaSAndroid Build Coastguard Worker         return true;
46*c8dee2aaSAndroid Build Coastguard Worker     } else {
47*c8dee2aaSAndroid Build Coastguard Worker         return (kRead_SkFILE_Flag == flags && ios_get_path_in_bundle(path, nullptr));
48*c8dee2aaSAndroid Build Coastguard Worker     }
49*c8dee2aaSAndroid Build Coastguard Worker #else
50*c8dee2aaSAndroid Build Coastguard Worker     return (0 == access(path, mode));
51*c8dee2aaSAndroid Build Coastguard Worker #endif
52*c8dee2aaSAndroid Build Coastguard Worker }
53*c8dee2aaSAndroid Build Coastguard Worker 
54*c8dee2aaSAndroid Build Coastguard Worker typedef struct {
55*c8dee2aaSAndroid Build Coastguard Worker     dev_t dev;
56*c8dee2aaSAndroid Build Coastguard Worker     ino_t ino;
57*c8dee2aaSAndroid Build Coastguard Worker } SkFILEID;
58*c8dee2aaSAndroid Build Coastguard Worker 
sk_ino(FILE * a,SkFILEID * id)59*c8dee2aaSAndroid Build Coastguard Worker static bool sk_ino(FILE* a, SkFILEID* id) {
60*c8dee2aaSAndroid Build Coastguard Worker     int fd = fileno(a);
61*c8dee2aaSAndroid Build Coastguard Worker     if (fd < 0) {
62*c8dee2aaSAndroid Build Coastguard Worker         return 0;
63*c8dee2aaSAndroid Build Coastguard Worker     }
64*c8dee2aaSAndroid Build Coastguard Worker     struct stat status = {};
65*c8dee2aaSAndroid Build Coastguard Worker     if (0 != fstat(fd, &status)) {
66*c8dee2aaSAndroid Build Coastguard Worker         return 0;
67*c8dee2aaSAndroid Build Coastguard Worker     }
68*c8dee2aaSAndroid Build Coastguard Worker     id->dev = status.st_dev;
69*c8dee2aaSAndroid Build Coastguard Worker     id->ino = status.st_ino;
70*c8dee2aaSAndroid Build Coastguard Worker     return true;
71*c8dee2aaSAndroid Build Coastguard Worker }
72*c8dee2aaSAndroid Build Coastguard Worker 
sk_fidentical(FILE * a,FILE * b)73*c8dee2aaSAndroid Build Coastguard Worker bool sk_fidentical(FILE* a, FILE* b) {
74*c8dee2aaSAndroid Build Coastguard Worker     SkFILEID aID, bID;
75*c8dee2aaSAndroid Build Coastguard Worker     return sk_ino(a, &aID) && sk_ino(b, &bID)
76*c8dee2aaSAndroid Build Coastguard Worker            && aID.ino == bID.ino
77*c8dee2aaSAndroid Build Coastguard Worker            && aID.dev == bID.dev;
78*c8dee2aaSAndroid Build Coastguard Worker }
79*c8dee2aaSAndroid Build Coastguard Worker 
sk_fmunmap(const void * addr,size_t length)80*c8dee2aaSAndroid Build Coastguard Worker void sk_fmunmap(const void* addr, size_t length) {
81*c8dee2aaSAndroid Build Coastguard Worker     munmap(const_cast<void*>(addr), length);
82*c8dee2aaSAndroid Build Coastguard Worker }
83*c8dee2aaSAndroid Build Coastguard Worker 
sk_fdmmap(int fd,size_t * size)84*c8dee2aaSAndroid Build Coastguard Worker void* sk_fdmmap(int fd, size_t* size) {
85*c8dee2aaSAndroid Build Coastguard Worker     struct stat status = {};
86*c8dee2aaSAndroid Build Coastguard Worker     if (0 != fstat(fd, &status)) {
87*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
88*c8dee2aaSAndroid Build Coastguard Worker     }
89*c8dee2aaSAndroid Build Coastguard Worker     if (!S_ISREG(status.st_mode)) {
90*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
91*c8dee2aaSAndroid Build Coastguard Worker     }
92*c8dee2aaSAndroid Build Coastguard Worker     if (!SkTFitsIn<size_t>(status.st_size)) {
93*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
94*c8dee2aaSAndroid Build Coastguard Worker     }
95*c8dee2aaSAndroid Build Coastguard Worker     size_t fileSize = static_cast<size_t>(status.st_size);
96*c8dee2aaSAndroid Build Coastguard Worker 
97*c8dee2aaSAndroid Build Coastguard Worker     void* addr = mmap(nullptr, fileSize, PROT_READ, MAP_PRIVATE, fd, 0);
98*c8dee2aaSAndroid Build Coastguard Worker     if (MAP_FAILED == addr) {
99*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
100*c8dee2aaSAndroid Build Coastguard Worker     }
101*c8dee2aaSAndroid Build Coastguard Worker 
102*c8dee2aaSAndroid Build Coastguard Worker     *size = fileSize;
103*c8dee2aaSAndroid Build Coastguard Worker     return addr;
104*c8dee2aaSAndroid Build Coastguard Worker }
105*c8dee2aaSAndroid Build Coastguard Worker 
sk_fileno(FILE * f)106*c8dee2aaSAndroid Build Coastguard Worker int sk_fileno(FILE* f) {
107*c8dee2aaSAndroid Build Coastguard Worker     return fileno(f);
108*c8dee2aaSAndroid Build Coastguard Worker }
109*c8dee2aaSAndroid Build Coastguard Worker 
sk_fmmap(FILE * f,size_t * size)110*c8dee2aaSAndroid Build Coastguard Worker void* sk_fmmap(FILE* f, size_t* size) {
111*c8dee2aaSAndroid Build Coastguard Worker     int fd = sk_fileno(f);
112*c8dee2aaSAndroid Build Coastguard Worker     if (fd < 0) {
113*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
114*c8dee2aaSAndroid Build Coastguard Worker     }
115*c8dee2aaSAndroid Build Coastguard Worker 
116*c8dee2aaSAndroid Build Coastguard Worker     return sk_fdmmap(fd, size);
117*c8dee2aaSAndroid Build Coastguard Worker }
118*c8dee2aaSAndroid Build Coastguard Worker 
sk_qread(FILE * file,void * buffer,size_t count,size_t offset)119*c8dee2aaSAndroid Build Coastguard Worker size_t sk_qread(FILE* file, void* buffer, size_t count, size_t offset) {
120*c8dee2aaSAndroid Build Coastguard Worker     int fd = sk_fileno(file);
121*c8dee2aaSAndroid Build Coastguard Worker     if (fd < 0) {
122*c8dee2aaSAndroid Build Coastguard Worker         return SIZE_MAX;
123*c8dee2aaSAndroid Build Coastguard Worker     }
124*c8dee2aaSAndroid Build Coastguard Worker     ssize_t bytesRead = pread(fd, buffer, count, offset);
125*c8dee2aaSAndroid Build Coastguard Worker     if (bytesRead < 0) {
126*c8dee2aaSAndroid Build Coastguard Worker         return SIZE_MAX;
127*c8dee2aaSAndroid Build Coastguard Worker     }
128*c8dee2aaSAndroid Build Coastguard Worker     return bytesRead;
129*c8dee2aaSAndroid Build Coastguard Worker }
130*c8dee2aaSAndroid Build Coastguard Worker 
131*c8dee2aaSAndroid Build Coastguard Worker ////////////////////////////////////////////////////////////////////////////
132*c8dee2aaSAndroid Build Coastguard Worker 
133*c8dee2aaSAndroid Build Coastguard Worker struct SkOSFileIterData {
SkOSFileIterDataSkOSFileIterData134*c8dee2aaSAndroid Build Coastguard Worker     SkOSFileIterData() : fDIR(nullptr) { }
135*c8dee2aaSAndroid Build Coastguard Worker     DIR* fDIR;
136*c8dee2aaSAndroid Build Coastguard Worker     SkString fPath, fSuffix;
137*c8dee2aaSAndroid Build Coastguard Worker };
138*c8dee2aaSAndroid Build Coastguard Worker static_assert(sizeof(SkOSFileIterData) <= SkOSFile::Iter::kStorageSize, "not_enough_space");
139*c8dee2aaSAndroid Build Coastguard Worker 
Iter()140*c8dee2aaSAndroid Build Coastguard Worker SkOSFile::Iter::Iter() { new (fSelf) SkOSFileIterData; }
141*c8dee2aaSAndroid Build Coastguard Worker 
Iter(const char path[],const char suffix[])142*c8dee2aaSAndroid Build Coastguard Worker SkOSFile::Iter::Iter(const char path[], const char suffix[]) {
143*c8dee2aaSAndroid Build Coastguard Worker     new (fSelf) SkOSFileIterData;
144*c8dee2aaSAndroid Build Coastguard Worker     this->reset(path, suffix);
145*c8dee2aaSAndroid Build Coastguard Worker }
146*c8dee2aaSAndroid Build Coastguard Worker 
~Iter()147*c8dee2aaSAndroid Build Coastguard Worker SkOSFile::Iter::~Iter() {
148*c8dee2aaSAndroid Build Coastguard Worker     SkOSFileIterData& self = *reinterpret_cast<SkOSFileIterData*>(fSelf);
149*c8dee2aaSAndroid Build Coastguard Worker     if (self.fDIR) {
150*c8dee2aaSAndroid Build Coastguard Worker         ::closedir(self.fDIR);
151*c8dee2aaSAndroid Build Coastguard Worker     }
152*c8dee2aaSAndroid Build Coastguard Worker     self.~SkOSFileIterData();
153*c8dee2aaSAndroid Build Coastguard Worker }
154*c8dee2aaSAndroid Build Coastguard Worker 
reset(const char path[],const char suffix[])155*c8dee2aaSAndroid Build Coastguard Worker void SkOSFile::Iter::reset(const char path[], const char suffix[]) {
156*c8dee2aaSAndroid Build Coastguard Worker     SkOSFileIterData& self = *reinterpret_cast<SkOSFileIterData*>(fSelf);
157*c8dee2aaSAndroid Build Coastguard Worker     if (self.fDIR) {
158*c8dee2aaSAndroid Build Coastguard Worker         ::closedir(self.fDIR);
159*c8dee2aaSAndroid Build Coastguard Worker         self.fDIR = nullptr;
160*c8dee2aaSAndroid Build Coastguard Worker     }
161*c8dee2aaSAndroid Build Coastguard Worker     self.fPath.set(path);
162*c8dee2aaSAndroid Build Coastguard Worker 
163*c8dee2aaSAndroid Build Coastguard Worker     if (path) {
164*c8dee2aaSAndroid Build Coastguard Worker         self.fDIR = ::opendir(path);
165*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_BUILD_FOR_IOS
166*c8dee2aaSAndroid Build Coastguard Worker         // check bundle for directory
167*c8dee2aaSAndroid Build Coastguard Worker         if (!self.fDIR && ios_get_path_in_bundle(path, &self.fPath)) {
168*c8dee2aaSAndroid Build Coastguard Worker             self.fDIR = ::opendir(self.fPath.c_str());
169*c8dee2aaSAndroid Build Coastguard Worker         }
170*c8dee2aaSAndroid Build Coastguard Worker #endif
171*c8dee2aaSAndroid Build Coastguard Worker         self.fSuffix.set(suffix);
172*c8dee2aaSAndroid Build Coastguard Worker     } else {
173*c8dee2aaSAndroid Build Coastguard Worker         self.fSuffix.reset();
174*c8dee2aaSAndroid Build Coastguard Worker     }
175*c8dee2aaSAndroid Build Coastguard Worker }
176*c8dee2aaSAndroid Build Coastguard Worker 
177*c8dee2aaSAndroid Build Coastguard Worker // returns true if suffix is empty, or if str ends with suffix
issuffixfor(const SkString & suffix,const char str[])178*c8dee2aaSAndroid Build Coastguard Worker static bool issuffixfor(const SkString& suffix, const char str[]) {
179*c8dee2aaSAndroid Build Coastguard Worker     size_t  suffixLen = suffix.size();
180*c8dee2aaSAndroid Build Coastguard Worker     size_t  strLen = strlen(str);
181*c8dee2aaSAndroid Build Coastguard Worker 
182*c8dee2aaSAndroid Build Coastguard Worker     return  strLen >= suffixLen &&
183*c8dee2aaSAndroid Build Coastguard Worker             memcmp(suffix.c_str(), str + strLen - suffixLen, suffixLen) == 0;
184*c8dee2aaSAndroid Build Coastguard Worker }
185*c8dee2aaSAndroid Build Coastguard Worker 
next(SkString * name,bool getDir)186*c8dee2aaSAndroid Build Coastguard Worker bool SkOSFile::Iter::next(SkString* name, bool getDir) {
187*c8dee2aaSAndroid Build Coastguard Worker     SkOSFileIterData& self = *reinterpret_cast<SkOSFileIterData*>(fSelf);
188*c8dee2aaSAndroid Build Coastguard Worker     if (self.fDIR) {
189*c8dee2aaSAndroid Build Coastguard Worker         dirent* entry;
190*c8dee2aaSAndroid Build Coastguard Worker 
191*c8dee2aaSAndroid Build Coastguard Worker         while ((entry = ::readdir(self.fDIR)) != nullptr) {
192*c8dee2aaSAndroid Build Coastguard Worker             struct stat s = {};
193*c8dee2aaSAndroid Build Coastguard Worker             SkString str(self.fPath);
194*c8dee2aaSAndroid Build Coastguard Worker 
195*c8dee2aaSAndroid Build Coastguard Worker             if (!str.endsWith("/") && !str.endsWith("\\")) {
196*c8dee2aaSAndroid Build Coastguard Worker                 str.append("/");
197*c8dee2aaSAndroid Build Coastguard Worker             }
198*c8dee2aaSAndroid Build Coastguard Worker             str.append(entry->d_name);
199*c8dee2aaSAndroid Build Coastguard Worker 
200*c8dee2aaSAndroid Build Coastguard Worker             if (0 == stat(str.c_str(), &s)) {
201*c8dee2aaSAndroid Build Coastguard Worker                 if (getDir) {
202*c8dee2aaSAndroid Build Coastguard Worker                     if (s.st_mode & S_IFDIR) {
203*c8dee2aaSAndroid Build Coastguard Worker                         break;
204*c8dee2aaSAndroid Build Coastguard Worker                     }
205*c8dee2aaSAndroid Build Coastguard Worker                 } else {
206*c8dee2aaSAndroid Build Coastguard Worker                     if (!(s.st_mode & S_IFDIR) && issuffixfor(self.fSuffix, entry->d_name)) {
207*c8dee2aaSAndroid Build Coastguard Worker                         break;
208*c8dee2aaSAndroid Build Coastguard Worker                     }
209*c8dee2aaSAndroid Build Coastguard Worker                 }
210*c8dee2aaSAndroid Build Coastguard Worker             }
211*c8dee2aaSAndroid Build Coastguard Worker         }
212*c8dee2aaSAndroid Build Coastguard Worker         if (entry) { // we broke out with a file
213*c8dee2aaSAndroid Build Coastguard Worker             if (name) {
214*c8dee2aaSAndroid Build Coastguard Worker                 name->set(entry->d_name);
215*c8dee2aaSAndroid Build Coastguard Worker             }
216*c8dee2aaSAndroid Build Coastguard Worker             return true;
217*c8dee2aaSAndroid Build Coastguard Worker         }
218*c8dee2aaSAndroid Build Coastguard Worker     }
219*c8dee2aaSAndroid Build Coastguard Worker     return false;
220*c8dee2aaSAndroid Build Coastguard Worker }
221