1 /**
2  * Cache the contents from files read from disk into memory.
3  *
4  * Copyright:   Copyright (C) 1999-2021 by The D Language Foundation, All Rights Reserved
5  * Authors:     $(LINK2 http://www.digitalmars.com, Walter Bright)
6  * License:     $(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost License 1.0)
7  * Source:      $(LINK2 https://github.com/dlang/dmd/blob/master/src/dmd/filecache.d, filecache.d)
8  * Documentation:  https://dlang.org/phobos/dmd_filecache.html
9  * Coverage:    https://codecov.io/gh/dlang/dmd/src/master/src/dmd/filecache.d
10  */
11 
12 module dmd.filecache;
13 
14 import dmd.root.stringtable;
15 import dmd.root.array;
16 import dmd.root.file;
17 import dmd.root.filename;
18 
19 import core.stdc.stdio;
20 
21 /**
22 A line-by-line representation of a $(REF File, dmd,root,file).
23 */
24 class FileAndLines
25 {
26     FileName* file;
27     FileBuffer* buffer;
28     const(char[])[] lines;
29 
30   nothrow:
31 
32     /**
33     File to read and split into its lines.
34     */
35     this(const(char)[] filename)
36     {
37         file = new FileName(filename);
38         readAndSplit();
39     }
40 
41     // Read a file and split the file buffer linewise
42     private void readAndSplit()
43     {
44         auto readResult = File.read(file.toChars());
45         // FIXME: check success
46         // take ownership of buffer
47         buffer = new FileBuffer(readResult.extractSlice());
48         ubyte* buf = buffer.data.ptr;
49         // slice into lines
50         while (*buf)
51         {
52             auto prevBuf = buf;
53             for (; *buf != '\n' && *buf != '\r'; buf++)
54             {
55                 if (!*buf)
56                     break;
57             }
58             // handle Windows line endings
59             if (*buf == '\r' && *(buf + 1) == '\n')
60                 buf++;
61             lines ~= cast(const(char)[]) prevBuf[0 .. buf - prevBuf];
62             buf++;
63         }
64     }
65 
66     void destroy()
67     {
68         if (file)
69         {
70             file.destroy();
71             file = null;
72             buffer.destroy();
73             buffer = null;
74             lines.destroy();
75             lines = null;
76         }
77     }
78 
79     ~this()
80     {
81         destroy();
82     }
83 }
84 
85 /**
86 A simple file cache that can be used to avoid reading the same file multiple times.
87 It stores its cached files as $(LREF FileAndLines)
88 */
89 struct FileCache
90 {
91     private StringTable!(FileAndLines) files;
92 
93   nothrow:
94 
95     /**
96     Add or get a file from the file cache.
97     If the file isn't part of the cache, it will be read from the filesystem.
98     If the file has been read before, the cached file object will be returned
99 
100     Params:
101         file = file to load in (or get from) the cache
102 
103     Returns: a $(LREF FileAndLines) object containing a line-by-line representation of the requested file
104     */
105     FileAndLines addOrGetFile(const(char)[] file)
106     {
107         if (auto payload = files.lookup(file))
108         {
109             if (payload !is null)
110                 return payload.value;
111         }
112 
113         auto lines = new FileAndLines(file);
114         files.insert(file, lines);
115         return lines;
116     }
117 
118     __gshared fileCache = FileCache();
119 
120     // Initializes the global FileCache singleton
121     static __gshared void _init()
122     {
123         fileCache.initialize();
124     }
125 
126     void initialize()
127     {
128         files._init();
129     }
130 
131     void deinitialize()
132     {
133         foreach (sv; files)
134             sv.destroy();
135         files.reset();
136     }
137 }