1 /*
2  * Copyright 2013 Google Inc.
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 
8 #if defined(SK_BUILD_FOR_MAC) || defined(SK_BUILD_FOR_UNIX) || defined(SK_BUILD_FOR_ANDROID)
9 #   include <unistd.h>
10 #   include <sys/time.h>
11 #   include <dirent.h>
12 #endif
13 
14 #if defined(SK_BUILD_FOR_MAC) || defined(SK_BUILD_FOR_UNIX)
15 #   include <glob.h>
16 #endif
17 
18 #if defined(SK_BUILD_FOR_MAC)
19 #   include <sys/syslimits.h> // PATH_MAX is here for Macs
20 #endif
21 
22 #if defined(SK_BUILD_FOR_WIN32)
23 #   include <windows.h>
24 #endif
25 
26 #include <stdlib.h>
27 #include <time.h>
28 #include "SkOSFile.h"
29 #include "skpdiff_util.h"
30 
31 #if SK_SUPPORT_OPENCL
cl_error_to_string(cl_int err)32 const char* cl_error_to_string(cl_int err) {
33     switch (err) {
34         case CL_SUCCESS:                         return "CL_SUCCESS";
35         case CL_DEVICE_NOT_FOUND:                return "CL_DEVICE_NOT_FOUND";
36         case CL_DEVICE_NOT_AVAILABLE:            return "CL_DEVICE_NOT_AVAILABLE";
37         case CL_COMPILER_NOT_AVAILABLE:          return "CL_COMPILER_NOT_AVAILABLE";
38         case CL_MEM_OBJECT_ALLOCATION_FAILURE:   return "CL_MEM_OBJECT_ALLOCATION_FAILURE";
39         case CL_OUT_OF_RESOURCES:                return "CL_OUT_OF_RESOURCES";
40         case CL_OUT_OF_HOST_MEMORY:              return "CL_OUT_OF_HOST_MEMORY";
41         case CL_PROFILING_INFO_NOT_AVAILABLE:    return "CL_PROFILING_INFO_NOT_AVAILABLE";
42         case CL_MEM_COPY_OVERLAP:                return "CL_MEM_COPY_OVERLAP";
43         case CL_IMAGE_FORMAT_MISMATCH:           return "CL_IMAGE_FORMAT_MISMATCH";
44         case CL_IMAGE_FORMAT_NOT_SUPPORTED:      return "CL_IMAGE_FORMAT_NOT_SUPPORTED";
45         case CL_BUILD_PROGRAM_FAILURE:           return "CL_BUILD_PROGRAM_FAILURE";
46         case CL_MAP_FAILURE:                     return "CL_MAP_FAILURE";
47         case CL_INVALID_VALUE:                   return "CL_INVALID_VALUE";
48         case CL_INVALID_DEVICE_TYPE:             return "CL_INVALID_DEVICE_TYPE";
49         case CL_INVALID_PLATFORM:                return "CL_INVALID_PLATFORM";
50         case CL_INVALID_DEVICE:                  return "CL_INVALID_DEVICE";
51         case CL_INVALID_CONTEXT:                 return "CL_INVALID_CONTEXT";
52         case CL_INVALID_QUEUE_PROPERTIES:        return "CL_INVALID_QUEUE_PROPERTIES";
53         case CL_INVALID_COMMAND_QUEUE:           return "CL_INVALID_COMMAND_QUEUE";
54         case CL_INVALID_HOST_PTR:                return "CL_INVALID_HOST_PTR";
55         case CL_INVALID_MEM_OBJECT:              return "CL_INVALID_MEM_OBJECT";
56         case CL_INVALID_IMAGE_FORMAT_DESCRIPTOR: return "CL_INVALID_IMAGE_FORMAT_DESCRIPTOR";
57         case CL_INVALID_IMAGE_SIZE:              return "CL_INVALID_IMAGE_SIZE";
58         case CL_INVALID_SAMPLER:                 return "CL_INVALID_SAMPLER";
59         case CL_INVALID_BINARY:                  return "CL_INVALID_BINARY";
60         case CL_INVALID_BUILD_OPTIONS:           return "CL_INVALID_BUILD_OPTIONS";
61         case CL_INVALID_PROGRAM:                 return "CL_INVALID_PROGRAM";
62         case CL_INVALID_PROGRAM_EXECUTABLE:      return "CL_INVALID_PROGRAM_EXECUTABLE";
63         case CL_INVALID_KERNEL_NAME:             return "CL_INVALID_KERNEL_NAME";
64         case CL_INVALID_KERNEL_DEFINITION:       return "CL_INVALID_KERNEL_DEFINITION";
65         case CL_INVALID_KERNEL:                  return "CL_INVALID_KERNEL";
66         case CL_INVALID_ARG_INDEX:               return "CL_INVALID_ARG_INDEX";
67         case CL_INVALID_ARG_VALUE:               return "CL_INVALID_ARG_VALUE";
68         case CL_INVALID_ARG_SIZE:                return "CL_INVALID_ARG_SIZE";
69         case CL_INVALID_KERNEL_ARGS:             return "CL_INVALID_KERNEL_ARGS";
70         case CL_INVALID_WORK_DIMENSION:          return "CL_INVALID_WORK_DIMENSION";
71         case CL_INVALID_WORK_GROUP_SIZE:         return "CL_INVALID_WORK_GROUP_SIZE";
72         case CL_INVALID_WORK_ITEM_SIZE:          return "CL_INVALID_WORK_ITEM_SIZE";
73         case CL_INVALID_GLOBAL_OFFSET:           return "CL_INVALID_GLOBAL_OFFSET";
74         case CL_INVALID_EVENT_WAIT_LIST:         return "CL_INVALID_EVENT_WAIT_LIST";
75         case CL_INVALID_EVENT:                   return "CL_INVALID_EVENT";
76         case CL_INVALID_OPERATION:               return "CL_INVALID_OPERATION";
77         case CL_INVALID_GL_OBJECT:               return "CL_INVALID_GL_OBJECT";
78         case CL_INVALID_BUFFER_SIZE:             return "CL_INVALID_BUFFER_SIZE";
79         case CL_INVALID_MIP_LEVEL:               return "CL_INVALID_MIP_LEVEL";
80         default:                                 return "UNKNOWN";
81     }
82     return "UNKNOWN";
83 }
84 #endif
85 
86 // TODO refactor Timer to be used here
get_seconds()87 double get_seconds() {
88 #if defined(SK_BUILD_FOR_WIN32)
89     LARGE_INTEGER currentTime;
90     LARGE_INTEGER frequency;
91     QueryPerformanceCounter(&currentTime);
92     QueryPerformanceFrequency(&frequency);
93     return (double)currentTime.QuadPart / (double)frequency.QuadPart;
94 #elif _POSIX_TIMERS > 0 && defined(CLOCK_REALTIME)
95     struct timespec currentTime;
96     clock_gettime(CLOCK_REALTIME, &currentTime);
97     return currentTime.tv_sec + (double)currentTime.tv_nsec / 1e9;
98 #elif defined(SK_BUILD_FOR_MAC) || defined(SK_BUILD_FOR_UNIX) || defined(SK_BUILD_FOR_ANDROID)
99     struct timeval currentTime;
100     gettimeofday(&currentTime, NULL);
101     return currentTime.tv_sec + (double)currentTime.tv_usec / 1e6;
102 #else
103     return clock() / (double)CLOCKS_PER_SEC;
104 #endif
105 }
106 
get_directory(const char path[],SkTArray<SkString> * entries)107 bool get_directory(const char path[], SkTArray<SkString>* entries) {
108 #if defined(SK_BUILD_FOR_MAC) || defined(SK_BUILD_FOR_UNIX) || defined(SK_BUILD_FOR_ANDROID)
109     // Open the directory and check for success
110     DIR* dir = opendir(path);
111     if (NULL == dir) {
112         return false;
113     }
114 
115     // Loop through dir entries until there are none left (i.e. readdir returns NULL)
116     struct dirent* entry;
117     while ((entry = readdir(dir))) {
118         // dirent only gives relative paths, we need to join them to the base path to check if they
119         // are directories.
120         SkString joinedPath = SkOSPath::Join(path, entry->d_name);
121 
122         // We only care about files
123         if (!sk_isdir(joinedPath.c_str())) {
124             entries->push_back(SkString(entry->d_name));
125         }
126     }
127 
128     closedir(dir);
129 
130     return true;
131 #elif defined(SK_BUILD_FOR_WIN32)
132     char pathDirGlob[MAX_PATH];
133     size_t pathLength = strlen(path);
134     strncpy(pathDirGlob, path, pathLength);
135 
136     if (path[pathLength - 1] == '/' || path[pathLength - 1] == '\\') {
137         SkASSERT(pathLength + 2 <= MAX_PATH);
138         pathDirGlob[pathLength] = '*';
139         pathDirGlob[pathLength + 1] = '\0';
140     } else {
141         SkASSERT(pathLength + 3 <= MAX_PATH);
142         pathDirGlob[pathLength] = '\\';
143         pathDirGlob[pathLength + 1] = '*';
144         pathDirGlob[pathLength + 2] = '\0';
145     }
146 
147     WIN32_FIND_DATA findFileData;
148     HANDLE hFind = FindFirstFile(pathDirGlob, &findFileData);
149     if (INVALID_HANDLE_VALUE == hFind) {
150         return false;
151     }
152 
153     do {
154         if ((findFileData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) == 0) {
155             entries->push_back(SkString(findFileData.cFileName));
156         }
157     } while (FindNextFile(hFind, &findFileData) != 0);
158 
159     FindClose(hFind);
160     return true;
161 #else
162     return false;
163 #endif
164 }
165 
glob_files(const char globPattern[],SkTArray<SkString> * entries)166 bool glob_files(const char globPattern[], SkTArray<SkString>* entries) {
167 #if defined(SK_BUILD_FOR_MAC) || defined(SK_BUILD_FOR_UNIX)
168     // TODO Make sure this works on windows. This may require use of FindNextFile windows function.
169     glob_t globBuffer;
170     if (glob(globPattern, 0, NULL, &globBuffer) != 0) {
171         return false;
172     }
173 
174     // Note these paths are in sorted order by default according to http://linux.die.net/man/3/glob
175     // Check under the flag GLOB_NOSORT
176     char** paths = globBuffer.gl_pathv;
177     while(*paths) {
178         entries->push_back(SkString(*paths));
179         paths++;
180     }
181 
182     globfree(&globBuffer);
183 
184     return true;
185 #else
186     return false;
187 #endif
188 }
189 
get_absolute_path(const SkString & path)190 SkString get_absolute_path(const SkString& path) {
191 #if defined(SK_BUILD_FOR_MAC) || defined(SK_BUILD_FOR_UNIX) || defined(SK_BUILD_FOR_ANDROID)
192     SkString fullPath(PATH_MAX + 1);
193     if (realpath(path.c_str(), fullPath.writable_str()) == NULL) {
194         fullPath.reset();
195     }
196     return fullPath;
197 #elif defined(SK_BUILD_FOR_WIN32)
198     SkString fullPath(MAX_PATH);
199     if (_fullpath(fullPath.writable_str(), path.c_str(), MAX_PATH) == NULL) {
200         fullPath.reset();
201     }
202     return fullPath;
203 #else
204     return SkString();
205 #endif
206 }
207