forked from svn2github/freearc
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathFiles.hs
613 lines (499 loc) · 24.8 KB
/
Files.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
{-# OPTIONS_GHC -cpp #-}
----------------------------------------------------------------------------------------------------
---- Îïåðàöèè ñ èìåíàìè ôàéëîâ, ìàíèïóëÿöèè ñ ôàéëàìè íà äèñêå, ââîä/âûâîä. ----
----------------------------------------------------------------------------------------------------
-----------------------------------------------------------------------------
-- |
-- Module : Files
-- Copyright : (c) Bulat Ziganshin <Bulat.Ziganshin@gmail.com>
-- License : Public domain
--
-- Maintainer : Bulat.Ziganshin@gmail.com
-- Stability : experimental
-- Portability : GHC
--
-----------------------------------------------------------------------------
module Files (module Files, module FilePath) where
import Prelude hiding (catch)
import Control.Concurrent
import Control.Concurrent.MVar
import Control.Exception
import Control.Monad
import Data.Array
import Data.Char
import Data.IORef
import Data.List
import Foreign
import Foreign.C
import Foreign.Marshal.Alloc
import System.Posix.Internals
import System.Posix.Types
import System.IO
import System.IO.Error hiding (catch)
import System.IO.Unsafe
import System.Environment
import System.Locale
import System.Time
import System.Process
import System.Directory
import Utils
import FilePath
#if defined(FREEARC_WIN)
import Win32Files
import System.Win32
#else
import System.Posix.Files hiding (fileExist)
#endif
-- |Ðàçìåð îäíîãî áóôåðà, èñïîëüçóåìûé â ðàçëè÷íûõ îïåðàöèÿõ
aBUFFER_SIZE = 64*kb
-- |Êîëè÷åñòâî áàéò, êîòîðûå äîëæíû ÷èòàòüñÿ/çàïèñûâàòüñÿ çà îäèí ðàç â áûñòðûõ ìåòîäàõ è ïðè ðàñïàêîâêå àñèììåòðè÷íûõ àëãîðèòìîâ
aLARGE_BUFFER_SIZE = 256*kb
-- |Êîëè÷åñòâî áàéò, êîòîðûå äîëæíû ÷èòàòüñÿ/çàïèñûâàòüñÿ çà îäèí ðàç â î÷åíü áûñòðûõ ìåòîäàõ (storing, tornado è òîìó ïîäîáíîå)
-- Ýòîò îáú¸ì ìèíèìèçèðóåò ïîòåðè íà disk seek operations - ïðè óñëîâèè, ÷òî îäíîâðåìåííî íå ïðîèñõîäèò â/â â äðóãîì ïîòîêå ;)
aHUGE_BUFFER_SIZE = 8*mb
----------------------------------------------------------------------------------------------------
---- Filename manipulations ------------------------------------------------------------------------
----------------------------------------------------------------------------------------------------
-- |True, åñëè file íàõîäèòñÿ â êàòàëîãå `dir`, îäíîì èç åãî ïîäêàòàëîãîâ, èëè ñîâïàäàåò ñ íèì
dir `isParentDirOf` file =
case (startFrom dir file) of
Just "" -> True
Just (x:_) -> isPathSeparator x
Nothing -> False
-- |Èìÿ ôàéëà çà ìèíóñîì êàòàëîãà dir
file `dropParentDir` dir =
case (startFrom dir file) of
Just "" -> ""
Just (x:xs) | isPathSeparator x -> xs
_ -> error "Utils::dropParentDir: dir isn't prefix of file"
#if defined(FREEARC_WIN)
-- |Äëÿ case-insensitive ôàéëîâûõ ñèñòåì
filenameLower = strLower
#else
-- |Äëÿ case-sensitive ôàéëîâûõ ñèñòåì
filenameLower = id
#endif
-- |Return False for special filenames like "." and ".." - used to filtering results of getDirContents
exclude_special_names s = (s/=".") && (s/="..")
-- Strip "drive:/" at the beginning of absolute filename
stripRoot = dropDrive
-- |Replace all '\' with '/'
translatePath = map (\c -> if isPathSeparator c then '/' else c)
-- |Filename extension, "dir/name.ext" -> "ext"
getFileSuffix = snd . splitFilenameSuffix
splitFilenameSuffix str = (name, drop 1 ext)
where (name, ext) = splitExtension str
-- "foo/bar/xyzzy.ext" -> ("foo/bar", "xyzzy.ext")
splitDirFilename :: String -> (String,String)
splitDirFilename str = case splitFileName str of
x@([d,':',s], name) -> x -- îñòàâëÿåì ("c:\", name)
(dir, name) -> (dropTrailingPathSeparator dir, name)
-- "foo/bar/xyzzy.ext" -> ("foo/bar", "xyzzy", "ext")
splitFilename3 :: String -> (String,String,String)
splitFilename3 str
= let (dir, rest) = splitDirFilename str
(name, ext) = splitFilenameSuffix rest
in (dir, name, ext)
-- | Modify the base name.
updateBaseName :: (String->String) -> FilePath -> FilePath
updateBaseName f pth = dir </> f name <.> ext
where
(dir, name, ext) = splitFilename3 pth
----------------------------------------------------------------------------------------------------
---- Ïîèñê êîíôèã-ôàéëîâ ïðîãðàììû è SFX ìîäóëåé ---------------------------------------------------
----------------------------------------------------------------------------------------------------
-- |Íàéòè êîíôèã-ôàéë ñ çàäàííûì èìåíåì èëè âîçâðàòèòü ""
findFile = findName fileExist
findDir = findName dirExist
findName exist possibleFilePlaces cfgfilename = do
found <- possibleFilePlaces cfgfilename >>= Utils.filterM exist
case found of
x:xs -> return x
[] -> return ""
-- |Íàéòè êîíôèã-ôàéë ñ çàäàííûì èìåíåì èëè âîçâðàòèòü èìÿ äëÿ ñîçäàíèÿ íîâîãî ôàéëà
findOrCreateFile possibleFilePlaces cfgfilename = do
variants <- possibleFilePlaces cfgfilename
found <- Utils.filterM fileExist variants
case found of
x:xs -> return x
[] -> return (head variants)
#if defined(FREEARC_WIN)
-- Ïîä Windows âñå äîïîëíèòåëüíûå ôàéëû ïî óìîë÷àíèþ ëåæàò â îäíîì êàòàëîãå ñ ïðîãðàììîé
libraryFilePlaces = configFilePlaces
configFilePlaces filename = do -- dir1 <- getAppUserDataDirectory "FreeArc"
exe <- getExeName
return [-- dir1 </> filename,
takeDirectory exe </> filename]
-- |Èìÿ èñïîëíÿåìîãî ôàéëà ïðîãðàììû
getExeName = do
allocaBytes (long_path_size*4) $ \pOutPath -> do
c_GetExeName pOutPath (fromIntegral long_path_size*2) >>= peekCWString
foreign import ccall unsafe "Environment.h GetExeName"
c_GetExeName :: CWFilePath -> CInt -> IO CWFilePath
#else
-- |Ìåñòà äëÿ ïîèñêà êîíôèã-ôàéëîâ
configFilePlaces filename = do dir1 <- getAppUserDataDirectory "FreeArc"
return [dir1 </> filename
,"/etc/FreeArc" </> filename]
-- |Ìåñòà äëÿ ïîèñêà sfx-ìîäóëåé
libraryFilePlaces filename = return ["/usr/lib/FreeArc" </> filename
,"/usr/local/lib/FreeArc" </> filename]
-- |Èìÿ èñïîëíÿåìîãî ôàéëà ïðîãðàììû
getExeName = getProgName
#endif
-- |Get temporary files directory
getTempDir = c_GetTempDir >>= peekCFilePath
foreign import ccall safe "Environment.h GetTempDir"
c_GetTempDir :: IO CFilePath
-- |Set directory for temporary files
setTempDir dir = withCFilePath dir c_SetTempDir
foreign import ccall safe "Environment.h SetTempDir"
c_SetTempDir :: CFilePath -> IO ()
----------------------------------------------------------------------------------------------------
---- Çàïóñê âíåøíèõ ïðîãðàìì è ðàáîòà ñ Windows registry -------------------------------------------
----------------------------------------------------------------------------------------------------
-- |Çàïóñòèòü êîìàíäó ÷åðåç shell è âîçâðàòèòü å¸ stdout
runProgram cmd = do
(_, stdout, stderr, ph) <- runInteractiveCommand cmd
forkIO (hGetContents stderr >>= evaluate.length >> return ())
result <- hGetContents stdout
evaluate (length result)
waitForProcess ph
return result
-- |Execute file/command in the directory `curdir` optionally waiting until it finished
runFile = runIt c_RunFile
runCommand = runIt c_RunCommand
runIt c_run_it filename curdir wait_finish = do
withCFilePath filename $ \c_filename -> do
withCFilePath curdir $ \c_curdir -> do
c_run_it c_filename c_curdir (i$fromEnum wait_finish)
foreign import ccall safe "Environment.h RunFile"
c_RunFile :: CFilePath -> CFilePath -> CInt -> IO ()
foreign import ccall safe "Environment.h RunCommand"
c_RunCommand :: CFilePath -> CFilePath -> CInt -> IO ()
-- |Ñîñòàâèòü ñòðîêó êîìàíäû èç ñïèñêà ñòðîê àðãóìåíòîâ
unparseCommand = joinWith " " . map quote
#if defined(FREEARC_WIN)
-- |Îòêðûòü HKEY è ïðî÷èòàòü èç Registry çíà÷åíèå òèïà REG_SZ
registryGetStr root branch key =
bracket (regOpenKey root branch) regCloseKey
(\hk -> registryGetStringValue hk key)
-- |Ñîçäàòü HKEY è çàïèñàòü â Registry çíà÷åíèå òèïà REG_SZ
registrySetStr root branch key val =
bracket (regCreateKey root branch) regCloseKey
(\hk -> registrySetStringValue hk key val)
-- |Ïðî÷èòàòü èç Registry çíà÷åíèå òèïà REG_SZ
registryGetStringValue :: HKEY -> String -> IO (Maybe String)
registryGetStringValue hk key = do
(regQueryValue hk (Just key) >>== Just)
`catch` (\e -> return Nothing)
-- |Çàïèñàòü â Registry çíà÷åíèå òèïà REG_SZ
registrySetStringValue :: HKEY -> String -> String -> IO ()
registrySetStringValue hk key val =
withTString val $ \v ->
regSetValueEx hk key rEG_SZ v (length val*2)
-- |Óäàëèòü öåëóþ âåòêó èç Registry
registryDeleteTree :: HKEY -> String -> IO ()
registryDeleteTree key subkey = do
handle (\e -> return ()) $ do
withForeignPtr key $ \ p_key -> do
withTString subkey $ \ c_subkey -> do
failUnlessSuccess "registryDeleteTree" $ c_RegistryDeleteTree p_key c_subkey
foreign import ccall unsafe "Environment.h RegistryDeleteTree"
c_RegistryDeleteTree :: PKEY -> LPCTSTR -> IO ErrCode
#endif
#if defined(FREEARC_WIN)
-- |OS-specific thread id
foreign import stdcall unsafe "windows.h GetCurrentThreadId"
getOsThreadId :: IO DWORD
#else
foreign import stdcall unsafe "pthread.h pthread_self"
getOsThreadId :: IO Int
#endif
----------------------------------------------------------------------------------------------------
---- Îïåðàöèè ñ íåîòêðûòûìè ôàéëàìè è êàòàëîãàìè ---------------------------------------------------
----------------------------------------------------------------------------------------------------
#if defined(FREEARC_WIN)
-- |Ñïèñîê äèñêîâ â ñèñòåìå ñ èõ òèïàìè
getDrives = getLogicalDrives >>== unfoldr (\n -> Just (n `mod` 2, n `div` 2))
>>== zipWith (\c n -> n>0 &&& [c:":"]) ['A'..'Z']
>>== concat
>>= mapM (\d -> do t <- withCString d c_GetDriveType; return (d++"\t"++(driveTypes!!i t)))
driveTypes = (split ',' "???,???,Removable,Fixed,Network,CD/DVD,Ramdisk") ++ repeat "???"
foreign import stdcall unsafe "windows.h GetDriveTypeA"
c_GetDriveType :: LPCSTR -> IO CInt
#endif
-- |Create a hierarchy of directories
createDirectoryHierarchy :: FilePath -> IO ()
createDirectoryHierarchy dir0 = do
let dir = dropTrailingPathSeparator dir0
d = stripRoot dir
when (d/= "" && exclude_special_names d) $ do
unlessM (dirExist dir) $ do
createDirectoryHierarchy (takeDirectory dir)
dirCreate dir
-- |Ñîçäàòü íåäîñòàþùèå êàòàëîãè íà ïóòè ê ôàéëó
buildPathTo filename = createDirectoryHierarchy (takeDirectory filename)
-- |Return current directory
getCurrentDirectory = myCanonicalizePath "."
-- | Given path referring to a file or directory, returns a
-- canonicalized path, with the intent that two paths referring
-- to the same file\/directory will map to the same canonicalized
-- path. Note that it is impossible to guarantee that the
-- implication (same file\/dir \<=\> same canonicalizedPath) holds
-- in either direction: this function can make only a best-effort
-- attempt.
myCanonicalizePath :: FilePath -> IO FilePath
myCanonicalizePath fpath | isURL fpath = return fpath
| otherwise =
#if defined(FREEARC_WIN)
withCFilePath fpath $ \pInPath ->
allocaBytes (long_path_size*4) $ \pOutPath ->
alloca $ \ppFilePart ->
do c_GetFullPathName pInPath (fromIntegral long_path_size*2) pOutPath ppFilePart
peekCFilePath pOutPath >>== dropTrailingPathSeparator
foreign import stdcall unsafe "GetFullPathNameW"
c_GetFullPathName :: CWString
-> CInt
-> CWString
-> Ptr CWString
-> IO CInt
#else
withCFilePath fpath $ \pInPath ->
allocaBytes (long_path_size*4) $ \pOutPath ->
do c_realpath pInPath pOutPath
peekCFilePath pOutPath >>== dropTrailingPathSeparator
foreign import ccall unsafe "realpath"
c_realpath :: CString
-> CString
-> IO CString
#endif
-- |Ìàêñèìàëüíàÿ äëèíà èìåíè ôàéëà
long_path_size = i c_long_path_size :: Int
foreign import ccall unsafe "Environment.h long_path_size"
c_long_path_size :: CInt
#if defined(FREEARC_WIN)
-- |Clear file's Archive bit
clearArchiveBit filename = do
attr <- getFileAttributes filename
when (attr.&.fILE_ATTRIBUTE_ARCHIVE /= 0) $ do
setFileAttributes filename (attr - fILE_ATTRIBUTE_ARCHIVE)
-- |Clear all file's attributes (before deletion)
clearFileAttributes filename = do
setFileAttributes filename 0
#else
clearArchiveBit = doNothing
clearFileAttributes = doNothing
#endif
-- |Ìèíèìàëüíîå datetime, êîòîðîå òîëüêî ìîæåò áûòü ó ôàéëà. Ñîîòâåòñòâóåò 1 ÿíâàðÿ 1970 ã.
aMINIMAL_POSSIBLE_DATETIME = 0 :: CTime
-- |Get file's date/time
getFileDateTime filename = fileWithStatus "getFileDateTime" filename stat_mtime
-- |Set file's date/time
setFileDateTime filename datetime = withCFilePath filename (`c_SetFileDateTime` datetime)
foreign import ccall unsafe "Environment.h SetFileDateTime"
c_SetFileDateTime :: CFilePath -> CTime -> IO ()
-- |Ïðåáðàçîâàíèå CTime â ClockTime. Èñïîëüçóåòñÿ èíôîðìàöèÿ î âíóòðåííåì ïðåäñòàâëåíèè ClockTime â GHC!!!
convert_CTime_to_ClockTime ctime = TOD (realToInteger ctime) 0
where realToInteger = round . realToFrac :: Real a => a -> Integer
-- |Ïðåáðàçîâàíèå ClockTime â CTime
convert_ClockTime_to_CTime (TOD secs _) = i secs
-- |Òåêñòîâîå ïðåäñòàâëåíèå âðåìåíè
showtime format t = formatCalendarTime defaultTimeLocale format (unsafePerformIO (toCalendarTime t))
-- |Îòôîðìàòèðîâàòü CTime â ñòðîêó ñ ôîðìàòîì "%Y-%m-%d %H:%M:%S"
formatDateTime t = unsafePerformIO $ do
allocaBytes 100 $ \buf -> do
c_FormatDateTime buf 100 t
peekCString buf
foreign import ccall unsafe "Environment.h FormatDateTime"
c_FormatDateTime :: CString -> CInt -> CTime -> IO ()
#if defined(FREEARC_UNIX)
executeModes = [ownerExecuteMode, groupExecuteMode, otherExecuteMode]
removeFileModes a b = a `intersectFileModes` (complement b)
#endif
-- Wait a few seconds (no more than half-hour due to Int overflow!)
sleepSeconds secs = do let us = round (secs*1000000)
threadDelay us
----------------------------------------------------------------------------------------------------
---- Îïåðàöèè ñ îòêðûòûìè ôàéëàìè ------------------------------------------------------------------
----------------------------------------------------------------------------------------------------
--withMVar mvar action = bracket (takeMVar mvar) (putMVar mvar) action
liftMVar1 action mvar = withMVar mvar action
liftMVar2 action mvar x = withMVar mvar (\a -> action a x)
liftMVar3 action mvar x y = withMVar mvar (\a -> action a x y)
returnMVar action = action >>= newMVar
-- |Àðõèâíûé ôàéë, çàâîðà÷èâàåòñÿ â MVar äëÿ ðåàëèçàöèè ïàðàëëåëüíîãî äîñòóïà èç ðàçíûõ òðåäîâ êî âõîäíûì àðõèâàì
data Archive = Archive { archiveName :: FilePath
, archiveFile :: MVar File
}
archiveOpen name = do file <- fileOpen name >>= newMVar; return (Archive name file)
archiveCreate name = do file <- fileCreate name >>= newMVar; return (Archive name file)
archiveCreateRW name = do file <- fileCreateRW name >>= newMVar; return (Archive name file)
archiveGetPos = liftMVar1 fileGetPos . archiveFile
archiveGetSize = liftMVar1 fileGetSize . archiveFile
archiveSeek = liftMVar2 fileSeek . archiveFile
archiveRead = liftMVar2 fileRead . archiveFile
archiveReadBuf = liftMVar3 fileReadBuf . archiveFile
archiveWrite = liftMVar2 fileWrite . archiveFile
archiveWriteBuf = liftMVar3 fileWriteBuf . archiveFile
archiveClose = liftMVar1 fileClose . archiveFile
-- |Ñêîïèðîâàòü äàííûå èç îäíîãî àðõèâà â äðóãîé è çàòåì âîññòàíîâèòü ïîçèöèþ â èñõîäíîì àðõèâå
archiveCopyData srcarc pos size dstarc = do
withMVar (archiveFile srcarc) $ \srcfile ->
withMVar (archiveFile dstarc) $ \dstfile -> do
restorePos <- fileGetPos srcfile
fileSeek srcfile pos
fileCopyBytes srcfile size dstfile
fileSeek srcfile restorePos
-- |Ïðè ðàáîòå ñ îäíèì ôèçè÷åñêèì äèñêîì (íàèáîëåå ÷àñòûé âàðèàíò)
-- íåò ñìûñëà âûïîëíÿòü íåñêîëüêî I/O îïåðàöèé ïàðàëëåëüíî,
-- ïîýòîìó ìû èõ âñå ïðîâîäèì ÷åðåç "óãîëüíîå óøêî" îäíîé-åäèíñòâåííîé MVar
oneIOAtTime = unsafePerformIO$ newMVar "oneIOAtTime value"
fileReadBuf file buf size = withMVar oneIOAtTime $ \_ -> fileReadBufSimple file buf size
fileWriteBuf file buf size = withMVar oneIOAtTime $ \_ -> fileWriteBufSimple file buf size
----------------------------------------------------------------------------------------------------
---- URL access ------------------------------------------------------------------------------------
----------------------------------------------------------------------------------------------------
data File = FileOnDisk FileOnDisk | URL URL
fileOpen = choose0 fOpen url_open
fileCreate = choose0 fCreate (\_ -> err "url_create")
fileCreateRW = choose0 fCreateRW (\_ -> err "url_create_rw")
fileAppendText = choose0 fAppendText (\_ -> err "url_append_text")
fileGetPos = choose fGetPos (url_pos .>>==i)
fileGetSize = choose fGetSize (url_size .>>==i)
fileSeek = choose fSeek (\f p -> url_seek f (i p))
fileReadBufSimple = choose fReadBufSimple url_read
fileWriteBufSimple = choose fWriteBufSimple (\_ _ _ -> err "url_write")
fileFlush = choose fFlush (\_ -> err "url_flush")
fileClose = choose fClose url_close
-- |Ïðîâåðÿåò ñóùåñòâîâàíèå ôàéëà/URL
fileExist name | isURL name = do url <- withCString name url_open
url_close url
return (url/=nullPtr)
| otherwise = fExist name
-- |Ïðîâåðÿåò, ÿâëÿåòñÿ ëè èìÿ url
isURL name = "://" `isInfixOf` name
{-# NOINLINE choose0 #-}
choose0 onfile onurl name | isURL name = do url <- withCString name onurl
when (url==nullPtr) $ do
fail$ "Can't open url "++name --registerError$ CANT_OPEN_FILE name
return (URL url)
| otherwise = onfile name >>== FileOnDisk
choose _ onurl (URL url) = onurl url
choose onfile _ (FileOnDisk file) = onfile file
{-# NOINLINE err #-}
err s = fail$ s++" isn't implemented" --registerError$ GENERAL_ERROR ["0343 %1 isn't implemented", s]
type URL = Ptr ()
foreign import ccall safe "URL.h" url_setup_proxy :: Ptr CChar -> IO ()
foreign import ccall safe "URL.h" url_setup_bypass_list :: Ptr CChar -> IO ()
foreign import ccall safe "URL.h" url_open :: Ptr CChar -> IO URL
foreign import ccall safe "URL.h" url_pos :: URL -> IO Int64
foreign import ccall safe "URL.h" url_size :: URL -> IO Int64
foreign import ccall safe "URL.h" url_seek :: URL -> Int64 -> IO ()
foreign import ccall safe "URL.h" url_read :: URL -> Ptr a -> Int -> IO Int
foreign import ccall safe "URL.h" url_close :: URL -> IO ()
----------------------------------------------------------------------------------------------------
---- Ïîä Windows ìíå ïðèøëîñü ðåàëèçîâàòü áèáëèîòåêó â/â ñàìîìó äëÿ ïîääåðæêè ôàéëîâ >4Gb è Unicode èì¸í ôàéëîâ
----------------------------------------------------------------------------------------------------
#if defined(FREEARC_WIN)
type FileOnDisk = FD
type CFilePath = CWFilePath
type FileAttributes = FileAttributeOrFlag
withCFilePath = withCWFilePath
peekCFilePath = peekCWString
fOpen name = wopen name (read_flags .|. o_BINARY) 0o666
fCreate name = wopen name (write_flags .|. o_BINARY .|. o_TRUNC) 0o666
fCreateRW name = wopen name (rw_flags .|. o_BINARY .|. o_TRUNC) 0o666
fAppendText name = wopen name (append_flags) 0o666
fGetPos = wtell
fGetSize = wfilelength
fSeek file pos = wseek file pos sEEK_SET
fReadBufSimple = wread
fWriteBufSimple = wwrite
fFlush file = return ()
fClose = wclose
fExist = wDoesFileExist
fileRemove = wunlink
fileRename = wrename
fileWithStatus = wWithFileStatus
fileStdin = 0
stat_mode = wst_mode
stat_size = wst_size
stat_mtime = wst_mtime
dirCreate = wmkdir
dirExist = wDoesDirectoryExist
dirRemove = wrmdir
dirList dir = dirWildcardList (dir </> "*")
dirWildcardList wc = withList $ \list -> do
wfindfiles wc $ \find -> do
name <- w_find_name find
list <<= name
#else
type FileOnDisk = Handle
type CFilePath = CString
type FileAttributes = Int
withCFilePath s a = (`withCString` a) =<< str2filesystem s
peekCFilePath ptr = peekCString ptr >>= filesystem2str
fOpen = (`openBinaryFile` ReadMode ) =<<. str2filesystem
fCreate = (`openBinaryFile` WriteMode ) =<<. str2filesystem
fCreateRW = (`openBinaryFile` ReadWriteMode) =<<. str2filesystem
fAppendText = (`openFile` AppendMode ) =<<. str2filesystem
fGetPos = hTell
fGetSize = hFileSize
fSeek = (`hSeek` AbsoluteSeek)
fReadBufSimple = hGetBuf
fWriteBufSimple = hPutBuf
fFlush = hFlush
fClose = hClose
fExist = doesFileExist =<<. str2filesystem
fileGetStatus = getFileStatus =<<. str2filesystem
fileSetMode name mode= (`setFileMode` mode) =<< str2filesystem name
fileRemove name = removeFile =<< str2filesystem name
fileRename a b = do a1 <- str2filesystem a; b1 <- str2filesystem b; renameFile a1 b1
fileSetSize = hSetFileSize
fileStdin = stdin
stat_mode = st_mode
stat_size = st_size .>>== i
stat_mtime = st_mtime
dirCreate = createDirectory =<<. str2filesystem
dirExist = doesDirectoryExist =<<. str2filesystem
dirRemove = removeDirectory =<<. str2filesystem
dirList dir = str2filesystem dir >>= getDirectoryContents >>= mapM filesystem2str
dirWildcardList wc = dirList (takeDirectory wc) >>== filter (match$ takeFileName wc)
-- kidnapped from System.Directory :)))
fileWithStatus :: String -> FilePath -> (Ptr CStat -> IO a) -> IO a
fileWithStatus loc name f = do
modifyIOError (`ioeSetFileName` name) $
allocaBytes sizeof_stat $ \p ->
withCFilePath name $ \s -> do
throwErrnoIfMinus1Retry_ loc (c_stat s p)
f p
#endif
fileRead file size = allocaBytes size $ \buf -> do fileReadBuf file buf size; peekCStringLen (buf,size)
fileWrite file str = withCStringLen str $ \(buf,size) -> fileWriteBuf file buf size
fileGetBinary name = bracket (fileOpen name) fileClose (\file -> fileGetSize file >>= fileRead file.i)
filePutBinary name str = bracket (fileCreate name) fileClose (`fileWrite` str)
-- |Ñêîïèðîâàòü çàäàííîå êîëè÷åñòâî áàéò èç îäíîãî îòêðûòîãî ôàéëà â äðóãîé
fileCopyBytes srcfile size dstfile = do
allocaBytes aHUGE_BUFFER_SIZE $ \buf -> do -- èñïîëüçóåì `alloca`, ÷òîáû àâòîìàòè÷åñêè îñâîáîäèòü âûäåëåííûé áóôåð ïðè âûõîäå
doChunks size aHUGE_BUFFER_SIZE $ \bytes -> do -- Ñêîïèðîâàòü size áàéò êóñêàìè ïî aHUGE_BUFFER_SIZE
bytes <- fileReadBuf srcfile buf bytes -- Ïðîâåðèì, ÷òî ïðî÷èòàíî ðîâíî ñòîëüêî áàéò, ñêîëüêî çàòðåáîâàíî
fileWriteBuf dstfile buf bytes
-- |True, åñëè ñóùåñòâóåò ôàéë èëè êàòàëîã ñ çàäàííûì èìåíåì
fileOrDirExist f = mapM ($f) [fileExist, dirExist] >>== or
---------------------------------------------------------------------------------------------------
---- Ãëîáàëüíûå íàñòðîéêè ïåðåêîäèðîâêè äëÿ èñïîëüçîâàíèÿ â ãëóáîêî âëîæåííûõ ôóíêöèÿõ ------------
---------------------------------------------------------------------------------------------------
-- |Translate filename from filesystem to internal encoding
filesystem2str' = unsafePerformIO$ newIORef$ id -- 'id' means that inifiles can't have non-English names
filesystem2str s = val filesystem2str' >>== ($s)
-- |Translate filename from internal to filesystem encoding
str2filesystem' = unsafePerformIO$ newIORef$ id
str2filesystem s = val str2filesystem' >>== ($s)
---------------------------------------------------------------------------------------------------
---- Utility functions ----------------------------------------------------------------------------
---------------------------------------------------------------------------------------------------
foreign import ccall unsafe "string.h"
memset :: Ptr a -> Int -> CSize -> IO ()
foreign import ccall unsafe "Environment.h memxor"
memxor :: Ptr a -> Ptr a -> Int -> IO ()