The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

450 lines
13KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library - "Jules' Utility Class Extensions"
  4. Copyright 2004-7 by Raw Material Software ltd.
  5. ------------------------------------------------------------------------------
  6. JUCE can be redistributed and/or modified under the terms of the
  7. GNU General Public License, as published by the Free Software Foundation;
  8. either version 2 of the License, or (at your option) any later version.
  9. JUCE is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with JUCE; if not, visit www.gnu.org/licenses or write to the
  15. Free Software Foundation, Inc., 59 Temple Place, Suite 330,
  16. Boston, MA 02111-1307 USA
  17. ------------------------------------------------------------------------------
  18. If you'd like to release a closed-source product which uses JUCE, commercial
  19. licenses are also available: visit www.rawmaterialsoftware.com/juce for
  20. more information.
  21. ==============================================================================
  22. */
  23. #include "linuxincludes.h"
  24. #include "../../../src/juce_core/basics/juce_StandardHeader.h"
  25. #include <sys/stat.h>
  26. #include <sys/dir.h>
  27. #include <sys/ptrace.h>
  28. #include <sys/vfs.h> // for statfs
  29. #include <sys/wait.h>
  30. #include <unistd.h>
  31. #include <fnmatch.h>
  32. #include <utime.h>
  33. #include <pwd.h>
  34. #include <fcntl.h>
  35. #define U_ISOFS_SUPER_MAGIC (short) 0x9660 // linux/iso_fs.h
  36. #define U_MSDOS_SUPER_MAGIC (short) 0x4d44 // linux/msdos_fs.h
  37. #define U_NFS_SUPER_MAGIC (short) 0x6969 // linux/nfs_fs.h
  38. #define U_SMB_SUPER_MAGIC (short) 0x517B // linux/smb_fs.h
  39. BEGIN_JUCE_NAMESPACE
  40. #include "../../../src/juce_core/io/files/juce_FileInputStream.h"
  41. #include "../../../src/juce_core/io/files/juce_FileOutputStream.h"
  42. #include "../../../src/juce_core/basics/juce_SystemStats.h"
  43. #include "../../../src/juce_core/basics/juce_Time.h"
  44. #include "../../../src/juce_core/io/network/juce_URL.h"
  45. #include "../../../src/juce_core/io/files/juce_NamedPipe.h"
  46. #include "../../../src/juce_core/threads/juce_InterProcessLock.h"
  47. #include "../../../src/juce_core/threads/juce_Thread.h"
  48. //==============================================================================
  49. /*
  50. Note that a lot of methods that you'd expect to find in this file actually
  51. live in juce_posix_SharedCode.cpp!
  52. */
  53. #include "../../macosx/platform_specific_code/juce_posix_SharedCode.cpp"
  54. //==============================================================================
  55. static File executableFile;
  56. //==============================================================================
  57. void juce_getFileTimes (const String& fileName,
  58. int64& modificationTime,
  59. int64& accessTime,
  60. int64& creationTime) throw()
  61. {
  62. modificationTime = 0;
  63. accessTime = 0;
  64. creationTime = 0;
  65. struct stat info;
  66. const int res = stat (fileName.toUTF8(), &info);
  67. if (res == 0)
  68. {
  69. modificationTime = (int64) info.st_mtime * 1000;
  70. accessTime = (int64) info.st_atime * 1000;
  71. creationTime = (int64) info.st_ctime * 1000;
  72. }
  73. }
  74. bool juce_setFileTimes (const String& fileName,
  75. int64 modificationTime,
  76. int64 accessTime,
  77. int64 creationTime) throw()
  78. {
  79. struct utimbuf times;
  80. times.actime = (time_t) (accessTime / 1000);
  81. times.modtime = (time_t) (modificationTime / 1000);
  82. return utime (fileName.toUTF8(), &times) == 0;
  83. }
  84. bool juce_setFileReadOnly (const String& fileName, bool isReadOnly) throw()
  85. {
  86. struct stat info;
  87. const int res = stat (fileName.toUTF8(), &info);
  88. if (res != 0)
  89. return false;
  90. info.st_mode &= 0777; // Just permissions
  91. if( isReadOnly )
  92. info.st_mode &= ~(S_IWUSR | S_IWGRP | S_IWOTH);
  93. else
  94. // Give everybody write permission?
  95. info.st_mode |= S_IWUSR | S_IWGRP | S_IWOTH;
  96. return chmod (fileName.toUTF8(), info.st_mode) == 0;
  97. }
  98. bool juce_copyFile (const String& s, const String& d) throw()
  99. {
  100. const File source (s), dest (d);
  101. FileInputStream* in = source.createInputStream();
  102. bool ok = false;
  103. if (in != 0)
  104. {
  105. if (dest.deleteFile())
  106. {
  107. FileOutputStream* const out = dest.createOutputStream();
  108. if (out != 0)
  109. {
  110. const int bytesCopied = out->writeFromInputStream (*in, -1);
  111. delete out;
  112. ok = (bytesCopied == source.getSize());
  113. if (! ok)
  114. dest.deleteFile();
  115. }
  116. }
  117. delete in;
  118. }
  119. return ok;
  120. }
  121. const StringArray juce_getFileSystemRoots() throw()
  122. {
  123. StringArray s;
  124. s.add (T("/"));
  125. return s;
  126. }
  127. //==============================================================================
  128. bool File::isOnCDRomDrive() const throw()
  129. {
  130. struct statfs buf;
  131. if (statfs (getFullPathName().toUTF8(), &buf) == 0)
  132. return (buf.f_type == U_ISOFS_SUPER_MAGIC);
  133. // Assume not if this fails for some reason
  134. return false;
  135. }
  136. bool File::isOnHardDisk() const throw()
  137. {
  138. struct statfs buf;
  139. if (statfs (getFullPathName().toUTF8(), &buf) == 0)
  140. {
  141. switch (buf.f_type)
  142. {
  143. case U_ISOFS_SUPER_MAGIC: // CD-ROM
  144. case U_MSDOS_SUPER_MAGIC: // Probably floppy (but could be mounted FAT filesystem)
  145. case U_NFS_SUPER_MAGIC: // Network NFS
  146. case U_SMB_SUPER_MAGIC: // Network Samba
  147. return false;
  148. default:
  149. // Assume anything else is a hard-disk (but note it could
  150. // be a RAM disk. There isn't a good way of determining
  151. // this for sure)
  152. return true;
  153. }
  154. }
  155. // Assume so if this fails for some reason
  156. return true;
  157. }
  158. bool File::isHidden() const throw()
  159. {
  160. return getFileName().startsWithChar (T('.'));
  161. }
  162. //==============================================================================
  163. const File File::getSpecialLocation (const SpecialLocationType type)
  164. {
  165. switch (type)
  166. {
  167. case userHomeDirectory:
  168. {
  169. const char* homeDir = getenv ("HOME");
  170. if (homeDir == 0)
  171. {
  172. struct passwd* const pw = getpwuid (getuid());
  173. if (pw != 0)
  174. homeDir = pw->pw_dir;
  175. }
  176. return File (String::fromUTF8 ((const uint8*) homeDir));
  177. }
  178. case userDocumentsDirectory:
  179. case userMusicDirectory:
  180. case userMoviesDirectory:
  181. case userApplicationDataDirectory:
  182. return File ("~");
  183. case userDesktopDirectory:
  184. return File ("~/Desktop");
  185. case commonApplicationDataDirectory:
  186. return File ("/var");
  187. case globalApplicationsDirectory:
  188. return File ("/usr");
  189. case tempDirectory:
  190. {
  191. File tmp ("/var/tmp");
  192. if (! tmp.isDirectory())
  193. {
  194. tmp = T("/tmp");
  195. if (! tmp.isDirectory())
  196. tmp = File::getCurrentWorkingDirectory();
  197. }
  198. return tmp;
  199. }
  200. case currentExecutableFile:
  201. case currentApplicationFile:
  202. // if this fails, it's probably because juce_setCurrentExecutableFileName()
  203. // was never called to set the filename - this should be done by the juce
  204. // main() function, so maybe you've hacked it to use your own custom main()?
  205. jassert (executableFile.exists());
  206. return executableFile;
  207. default:
  208. jassertfalse // unknown type?
  209. break;
  210. }
  211. return File::nonexistent;
  212. }
  213. //==============================================================================
  214. void juce_setCurrentExecutableFileName (const String& filename) throw()
  215. {
  216. executableFile = File::getCurrentWorkingDirectory().getChildFile (filename);
  217. }
  218. //==============================================================================
  219. const File File::getCurrentWorkingDirectory() throw()
  220. {
  221. char buf [2048];
  222. getcwd (buf, sizeof(buf));
  223. return File (String::fromUTF8 ((const uint8*) buf));
  224. }
  225. bool File::setAsCurrentWorkingDirectory() const throw()
  226. {
  227. return chdir (getFullPathName().toUTF8()) == 0;
  228. }
  229. //==============================================================================
  230. struct FindFileStruct
  231. {
  232. String parentDir, wildCard;
  233. DIR* dir;
  234. bool getNextMatch (String& result, bool* const isDir, bool* const isHidden, int64* const fileSize,
  235. Time* const modTime, Time* const creationTime, bool* const isReadOnly) throw()
  236. {
  237. const char* const wildcardUTF8 = wildCard.toUTF8();
  238. for (;;)
  239. {
  240. struct dirent* const de = readdir (dir);
  241. if (de == 0)
  242. break;
  243. if (fnmatch (wildcardUTF8, de->d_name, FNM_CASEFOLD) == 0)
  244. {
  245. result = String::fromUTF8 ((const uint8*) de->d_name);
  246. const String path (parentDir + result);
  247. if (isDir != 0 || fileSize != 0)
  248. {
  249. struct stat info;
  250. const bool statOk = (stat (path.toUTF8(), &info) == 0);
  251. if (isDir != 0)
  252. *isDir = path.isEmpty() || (statOk && ((info.st_mode & S_IFDIR) != 0));
  253. if (isHidden != 0)
  254. *isHidden = (de->d_name[0] == '.');
  255. if (fileSize != 0)
  256. *fileSize = statOk ? info.st_size : 0;
  257. }
  258. if (modTime != 0 || creationTime != 0)
  259. {
  260. int64 m, a, c;
  261. juce_getFileTimes (path, m, a, c);
  262. if (modTime != 0)
  263. *modTime = m;
  264. if (creationTime != 0)
  265. *creationTime = c;
  266. }
  267. if (isReadOnly != 0)
  268. *isReadOnly = ! juce_canWriteToFile (path);
  269. return true;
  270. }
  271. }
  272. return false;
  273. }
  274. };
  275. // returns 0 on failure
  276. void* juce_findFileStart (const String& directory, const String& wildCard, String& firstResultFile,
  277. bool* isDir, bool* isHidden, int64* fileSize, Time* modTime,
  278. Time* creationTime, bool* isReadOnly) throw()
  279. {
  280. DIR* d = opendir (directory.toUTF8());
  281. if (d != 0)
  282. {
  283. FindFileStruct* ff = new FindFileStruct();
  284. ff->parentDir = directory;
  285. if (!ff->parentDir.endsWithChar (File::separator))
  286. ff->parentDir += File::separator;
  287. ff->wildCard = wildCard;
  288. if (wildCard == T("*.*"))
  289. ff->wildCard = T("*");
  290. ff->dir = d;
  291. if (ff->getNextMatch (firstResultFile, isDir, isHidden, fileSize, modTime, creationTime, isReadOnly))
  292. {
  293. return ff;
  294. }
  295. else
  296. {
  297. firstResultFile = String::empty;
  298. isDir = false;
  299. isHidden = false;
  300. closedir (d);
  301. delete ff;
  302. }
  303. }
  304. return 0;
  305. }
  306. bool juce_findFileNext (void* handle, String& resultFile,
  307. bool* isDir, bool* isHidden, int64* fileSize, Time* modTime, Time* creationTime, bool* isReadOnly) throw()
  308. {
  309. FindFileStruct* const ff = (FindFileStruct*) handle;
  310. if (ff != 0)
  311. return ff->getNextMatch (resultFile, isDir, isHidden, fileSize, modTime, creationTime, isReadOnly);
  312. return false;
  313. }
  314. void juce_findFileClose (void* handle) throw()
  315. {
  316. FindFileStruct* const ff = (FindFileStruct*) handle;
  317. if (ff != 0)
  318. {
  319. closedir (ff->dir);
  320. delete ff;
  321. }
  322. }
  323. bool juce_launchFile (const String& fileName,
  324. const String& parameters) throw()
  325. {
  326. String cmdString (fileName);
  327. cmdString << " " << parameters;
  328. if (URL::isProbablyAWebsiteURL (cmdString) || URL::isProbablyAnEmailAddress (cmdString))
  329. {
  330. // create a command that tries to launch a bunch of likely browsers
  331. const char* const browserNames[] = { "/etc/alternatives/x-www-browser", "firefox", "mozilla", "konqueror", "opera" };
  332. StringArray cmdLines;
  333. for (int i = 0; i < numElementsInArray (browserNames); ++i)
  334. cmdLines.add (String (browserNames[i]) + T(" ") + cmdString.trim().quoted());
  335. cmdString = cmdLines.joinIntoString (T(" || "));
  336. }
  337. char* const argv[4] = { "/bin/sh", "-c", (char*) cmdString.toUTF8(), 0 };
  338. const int cpid = fork();
  339. if (cpid == 0)
  340. {
  341. setsid();
  342. // Child process
  343. execve (argv[0], argv, environ);
  344. exit (0);
  345. }
  346. return cpid >= 0;
  347. }
  348. END_JUCE_NAMESPACE