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