• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2011 Google Inc. All Rights Reserved.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 //     http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #include <assert.h>
16 #include <stdio.h>
17 #ifdef _WIN32
18 #include <io.h>
19 #include <windows.h>
20 #endif
21 
22 #include "disk_interface.h"
23 #include "graph.h"
24 #include "test.h"
25 
26 namespace {
27 
28 struct DiskInterfaceTest : public testing::Test {
SetUp__anon72471dc80111::DiskInterfaceTest29   virtual void SetUp() {
30     // These tests do real disk accesses, so create a temp dir.
31     temp_dir_.CreateAndEnter("Ninja-DiskInterfaceTest");
32   }
33 
TearDown__anon72471dc80111::DiskInterfaceTest34   virtual void TearDown() {
35     temp_dir_.Cleanup();
36   }
37 
Touch__anon72471dc80111::DiskInterfaceTest38   bool Touch(const char* path) {
39     FILE *f = fopen(path, "w");
40     if (!f)
41       return false;
42     return fclose(f) == 0;
43   }
44 
45   ScopedTempDir temp_dir_;
46   RealDiskInterface disk_;
47 };
48 
TEST_F(DiskInterfaceTest,StatMissingFile)49 TEST_F(DiskInterfaceTest, StatMissingFile) {
50   string err;
51   EXPECT_EQ(0, disk_.Stat("nosuchfile", &err));
52   EXPECT_EQ("", err);
53 
54   // On Windows, the errno for a file in a nonexistent directory
55   // is different.
56   EXPECT_EQ(0, disk_.Stat("nosuchdir/nosuchfile", &err));
57   EXPECT_EQ("", err);
58 
59   // On POSIX systems, the errno is different if a component of the
60   // path prefix is not a directory.
61   ASSERT_TRUE(Touch("notadir"));
62   EXPECT_EQ(0, disk_.Stat("notadir/nosuchfile", &err));
63   EXPECT_EQ("", err);
64 }
65 
TEST_F(DiskInterfaceTest,StatBadPath)66 TEST_F(DiskInterfaceTest, StatBadPath) {
67   string err;
68 #ifdef _WIN32
69   string bad_path("cc:\\foo");
70   EXPECT_EQ(-1, disk_.Stat(bad_path, &err));
71   EXPECT_NE("", err);
72 #else
73   string too_long_name(512, 'x');
74   EXPECT_EQ(-1, disk_.Stat(too_long_name, &err));
75   EXPECT_NE("", err);
76 #endif
77 }
78 
TEST_F(DiskInterfaceTest,StatExistingFile)79 TEST_F(DiskInterfaceTest, StatExistingFile) {
80   string err;
81   ASSERT_TRUE(Touch("file"));
82   EXPECT_GT(disk_.Stat("file", &err), 1);
83   EXPECT_EQ("", err);
84 }
85 
TEST_F(DiskInterfaceTest,StatExistingDir)86 TEST_F(DiskInterfaceTest, StatExistingDir) {
87   string err;
88   ASSERT_TRUE(disk_.MakeDir("subdir"));
89   ASSERT_TRUE(disk_.MakeDir("subdir/subsubdir"));
90   EXPECT_GT(disk_.Stat("..", &err), 1);
91   EXPECT_EQ("", err);
92   EXPECT_GT(disk_.Stat(".", &err), 1);
93   EXPECT_EQ("", err);
94   EXPECT_GT(disk_.Stat("subdir", &err), 1);
95   EXPECT_EQ("", err);
96   EXPECT_GT(disk_.Stat("subdir/subsubdir", &err), 1);
97   EXPECT_EQ("", err);
98 
99   EXPECT_EQ(disk_.Stat("subdir", &err),
100             disk_.Stat("subdir/.", &err));
101   EXPECT_EQ(disk_.Stat("subdir", &err),
102             disk_.Stat("subdir/subsubdir/..", &err));
103   EXPECT_EQ(disk_.Stat("subdir/subsubdir", &err),
104             disk_.Stat("subdir/subsubdir/.", &err));
105 }
106 
107 #ifdef _WIN32
TEST_F(DiskInterfaceTest,StatCache)108 TEST_F(DiskInterfaceTest, StatCache) {
109   string err;
110 
111   ASSERT_TRUE(Touch("file1"));
112   ASSERT_TRUE(Touch("fiLE2"));
113   ASSERT_TRUE(disk_.MakeDir("subdir"));
114   ASSERT_TRUE(disk_.MakeDir("subdir/subsubdir"));
115   ASSERT_TRUE(Touch("subdir\\subfile1"));
116   ASSERT_TRUE(Touch("subdir\\SUBFILE2"));
117   ASSERT_TRUE(Touch("subdir\\SUBFILE3"));
118 
119   disk_.AllowStatCache(false);
120   TimeStamp parent_stat_uncached = disk_.Stat("..", &err);
121   disk_.AllowStatCache(true);
122 
123   EXPECT_GT(disk_.Stat("FIle1", &err), 1);
124   EXPECT_EQ("", err);
125   EXPECT_GT(disk_.Stat("file1", &err), 1);
126   EXPECT_EQ("", err);
127 
128   EXPECT_GT(disk_.Stat("subdir/subfile2", &err), 1);
129   EXPECT_EQ("", err);
130   EXPECT_GT(disk_.Stat("sUbdir\\suBFile1", &err), 1);
131   EXPECT_EQ("", err);
132 
133   EXPECT_GT(disk_.Stat("..", &err), 1);
134   EXPECT_EQ("", err);
135   EXPECT_GT(disk_.Stat(".", &err), 1);
136   EXPECT_EQ("", err);
137   EXPECT_GT(disk_.Stat("subdir", &err), 1);
138   EXPECT_EQ("", err);
139   EXPECT_GT(disk_.Stat("subdir/subsubdir", &err), 1);
140   EXPECT_EQ("", err);
141 
142 #ifndef _MSC_VER // TODO: Investigate why. Also see https://github.com/ninja-build/ninja/pull/1423
143   EXPECT_EQ(disk_.Stat("subdir", &err),
144             disk_.Stat("subdir/.", &err));
145   EXPECT_EQ("", err);
146   EXPECT_EQ(disk_.Stat("subdir", &err),
147             disk_.Stat("subdir/subsubdir/..", &err));
148 #endif
149   EXPECT_EQ("", err);
150   EXPECT_EQ(disk_.Stat("..", &err), parent_stat_uncached);
151   EXPECT_EQ("", err);
152   EXPECT_EQ(disk_.Stat("subdir/subsubdir", &err),
153             disk_.Stat("subdir/subsubdir/.", &err));
154   EXPECT_EQ("", err);
155 
156   // Test error cases.
157   string bad_path("cc:\\foo");
158   EXPECT_EQ(-1, disk_.Stat(bad_path, &err));
159   EXPECT_NE("", err); err.clear();
160   EXPECT_EQ(-1, disk_.Stat(bad_path, &err));
161   EXPECT_NE("", err); err.clear();
162   EXPECT_EQ(0, disk_.Stat("nosuchfile", &err));
163   EXPECT_EQ("", err);
164   EXPECT_EQ(0, disk_.Stat("nosuchdir/nosuchfile", &err));
165   EXPECT_EQ("", err);
166 }
167 #endif
168 
TEST_F(DiskInterfaceTest,ReadFile)169 TEST_F(DiskInterfaceTest, ReadFile) {
170   string err;
171   std::string content;
172   ASSERT_EQ(DiskInterface::NotFound,
173             disk_.ReadFile("foobar", &content, &err));
174   EXPECT_EQ("", content);
175   EXPECT_NE("", err); // actual value is platform-specific
176   err.clear();
177 
178   const char* kTestFile = "testfile";
179   FILE* f = fopen(kTestFile, "wb");
180   ASSERT_TRUE(f);
181   const char* kTestContent = "test content\nok";
182   fprintf(f, "%s", kTestContent);
183   ASSERT_EQ(0, fclose(f));
184 
185   ASSERT_EQ(DiskInterface::Okay,
186             disk_.ReadFile(kTestFile, &content, &err));
187   EXPECT_EQ(kTestContent, content);
188   EXPECT_EQ("", err);
189 }
190 
TEST_F(DiskInterfaceTest,MakeDirs)191 TEST_F(DiskInterfaceTest, MakeDirs) {
192   string path = "path/with/double//slash/";
193   EXPECT_TRUE(disk_.MakeDirs(path));
194   FILE* f = fopen((path + "a_file").c_str(), "w");
195   EXPECT_TRUE(f);
196   EXPECT_EQ(0, fclose(f));
197 #ifdef _WIN32
198   string path2 = "another\\with\\back\\\\slashes\\";
199   EXPECT_TRUE(disk_.MakeDirs(path2.c_str()));
200   FILE* f2 = fopen((path2 + "a_file").c_str(), "w");
201   EXPECT_TRUE(f2);
202   EXPECT_EQ(0, fclose(f2));
203 #endif
204 }
205 
TEST_F(DiskInterfaceTest,RemoveFile)206 TEST_F(DiskInterfaceTest, RemoveFile) {
207   const char* kFileName = "file-to-remove";
208   ASSERT_TRUE(Touch(kFileName));
209   EXPECT_EQ(0, disk_.RemoveFile(kFileName));
210   EXPECT_EQ(1, disk_.RemoveFile(kFileName));
211   EXPECT_EQ(1, disk_.RemoveFile("does not exist"));
212 }
213 
214 struct StatTest : public StateTestWithBuiltinRules,
215                   public DiskInterface {
StatTest__anon72471dc80111::StatTest216   StatTest() : scan_(&state_, NULL, NULL, this, NULL) {}
217 
218   // DiskInterface implementation.
219   virtual TimeStamp Stat(const string& path, string* err) const;
WriteFile__anon72471dc80111::StatTest220   virtual bool WriteFile(const string& path, const string& contents) {
221     assert(false);
222     return true;
223   }
MakeDir__anon72471dc80111::StatTest224   virtual bool MakeDir(const string& path) {
225     assert(false);
226     return false;
227   }
ReadFile__anon72471dc80111::StatTest228   virtual Status ReadFile(const string& path, string* contents, string* err) {
229     assert(false);
230     return NotFound;
231   }
RemoveFile__anon72471dc80111::StatTest232   virtual int RemoveFile(const string& path) {
233     assert(false);
234     return 0;
235   }
236 
237   DependencyScan scan_;
238   map<string, TimeStamp> mtimes_;
239   mutable vector<string> stats_;
240 };
241 
Stat(const string & path,string * err) const242 TimeStamp StatTest::Stat(const string& path, string* err) const {
243   stats_.push_back(path);
244   map<string, TimeStamp>::const_iterator i = mtimes_.find(path);
245   if (i == mtimes_.end())
246     return 0;  // File not found.
247   return i->second;
248 }
249 
TEST_F(StatTest,Simple)250 TEST_F(StatTest, Simple) {
251   ASSERT_NO_FATAL_FAILURE(AssertParse(&state_,
252 "build out: cat in\n"));
253 
254   Node* out = GetNode("out");
255   string err;
256   EXPECT_TRUE(out->Stat(this, &err));
257   EXPECT_EQ("", err);
258   ASSERT_EQ(1u, stats_.size());
259   scan_.RecomputeDirty(out, NULL);
260   ASSERT_EQ(2u, stats_.size());
261   ASSERT_EQ("out", stats_[0]);
262   ASSERT_EQ("in",  stats_[1]);
263 }
264 
TEST_F(StatTest,TwoStep)265 TEST_F(StatTest, TwoStep) {
266   ASSERT_NO_FATAL_FAILURE(AssertParse(&state_,
267 "build out: cat mid\n"
268 "build mid: cat in\n"));
269 
270   Node* out = GetNode("out");
271   string err;
272   EXPECT_TRUE(out->Stat(this, &err));
273   EXPECT_EQ("", err);
274   ASSERT_EQ(1u, stats_.size());
275   scan_.RecomputeDirty(out, NULL);
276   ASSERT_EQ(3u, stats_.size());
277   ASSERT_EQ("out", stats_[0]);
278   ASSERT_TRUE(GetNode("out")->dirty());
279   ASSERT_EQ("mid",  stats_[1]);
280   ASSERT_TRUE(GetNode("mid")->dirty());
281   ASSERT_EQ("in",  stats_[2]);
282 }
283 
TEST_F(StatTest,Tree)284 TEST_F(StatTest, Tree) {
285   ASSERT_NO_FATAL_FAILURE(AssertParse(&state_,
286 "build out: cat mid1 mid2\n"
287 "build mid1: cat in11 in12\n"
288 "build mid2: cat in21 in22\n"));
289 
290   Node* out = GetNode("out");
291   string err;
292   EXPECT_TRUE(out->Stat(this, &err));
293   EXPECT_EQ("", err);
294   ASSERT_EQ(1u, stats_.size());
295   scan_.RecomputeDirty(out, NULL);
296   ASSERT_EQ(1u + 6u, stats_.size());
297   ASSERT_EQ("mid1", stats_[1]);
298   ASSERT_TRUE(GetNode("mid1")->dirty());
299   ASSERT_EQ("in11", stats_[2]);
300 }
301 
TEST_F(StatTest,Middle)302 TEST_F(StatTest, Middle) {
303   ASSERT_NO_FATAL_FAILURE(AssertParse(&state_,
304 "build out: cat mid\n"
305 "build mid: cat in\n"));
306 
307   mtimes_["in"] = 1;
308   mtimes_["mid"] = 0;  // missing
309   mtimes_["out"] = 1;
310 
311   Node* out = GetNode("out");
312   string err;
313   EXPECT_TRUE(out->Stat(this, &err));
314   EXPECT_EQ("", err);
315   ASSERT_EQ(1u, stats_.size());
316   scan_.RecomputeDirty(out, NULL);
317   ASSERT_FALSE(GetNode("in")->dirty());
318   ASSERT_TRUE(GetNode("mid")->dirty());
319   ASSERT_TRUE(GetNode("out")->dirty());
320 }
321 
322 }  // namespace
323