1 //===----------------------------------------------------------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is dual licensed under the MIT and the University of Illinois Open
6 // Source Licenses. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 // UNSUPPORTED: c++98, c++03
11 
12 // <filesystem>
13 
14 // file_time_type last_write_time(const path& p);
15 // file_time_type last_write_time(const path& p, std::error_code& ec) noexcept;
16 // void last_write_time(const path& p, file_time_type new_time);
17 // void last_write_time(const path& p, file_time_type new_type,
18 //                      std::error_code& ec) noexcept;
19 
20 #include "filesystem_include.hpp"
21 #include <type_traits>
22 #include <chrono>
23 #include <fstream>
24 #include <cstdlib>
25 
26 #include "test_macros.h"
27 #include "rapid-cxx-test.hpp"
28 #include "filesystem_test_helper.hpp"
29 
30 #include <sys/stat.h>
31 #include <iostream>
32 
33 #include <fcntl.h>
34 #include <sys/time.h>
35 
36 using namespace fs;
37 
38 using TimeSpec = struct ::timespec;
39 using StatT = struct ::stat;
40 
41 using Sec = std::chrono::duration<file_time_type::rep>;
42 using Hours = std::chrono::hours;
43 using Minutes = std::chrono::minutes;
44 using MicroSec = std::chrono::duration<file_time_type::rep, std::micro>;
45 using NanoSec = std::chrono::duration<file_time_type::rep, std::nano>;
46 using std::chrono::duration_cast;
47 
48 #if defined(__APPLE__)
extract_mtime(StatT const & st)49 TimeSpec extract_mtime(StatT const& st) { return st.st_mtimespec; }
extract_atime(StatT const & st)50 TimeSpec extract_atime(StatT const& st) { return st.st_atimespec; }
51 #else
extract_mtime(StatT const & st)52 TimeSpec extract_mtime(StatT const& st) { return st.st_mtim; }
extract_atime(StatT const & st)53 TimeSpec extract_atime(StatT const& st) { return st.st_atim; }
54 #endif
55 
ConvertToTimeSpec(TimeSpec & ts,file_time_type ft)56 bool ConvertToTimeSpec(TimeSpec& ts, file_time_type ft) {
57   using SecFieldT = decltype(TimeSpec::tv_sec);
58   using NSecFieldT = decltype(TimeSpec::tv_nsec);
59   using SecLim = std::numeric_limits<SecFieldT>;
60   using NSecLim = std::numeric_limits<NSecFieldT>;
61 
62   auto secs = duration_cast<Sec>(ft.time_since_epoch());
63   auto nsecs = duration_cast<NanoSec>(ft.time_since_epoch() - secs);
64   if (nsecs.count() < 0) {
65     if (Sec::min().count() > SecLim::min()) {
66       secs += Sec(1);
67       nsecs -= Sec(1);
68     } else {
69       nsecs = NanoSec(0);
70     }
71   }
72   if (SecLim::max() < secs.count() || SecLim::min() > secs.count())
73     return false;
74   if (NSecLim::max() < nsecs.count() || NSecLim::min() > nsecs.count())
75     return false;
76   ts.tv_sec = secs.count();
77   ts.tv_nsec = nsecs.count();
78   return true;
79 }
80 
ConvertFromTimeSpec(file_time_type & ft,TimeSpec ts)81 bool ConvertFromTimeSpec(file_time_type& ft, TimeSpec ts) {
82   auto secs_part = duration_cast<file_time_type::duration>(Sec(ts.tv_sec));
83   if (duration_cast<Sec>(secs_part).count() != ts.tv_sec)
84     return false;
85   auto subsecs = duration_cast<file_time_type::duration>(NanoSec(ts.tv_nsec));
86   auto dur = secs_part + subsecs;
87   if (dur < secs_part && subsecs.count() >= 0)
88     return false;
89   ft = file_time_type(dur);
90   return true;
91 }
92 
CompareTimeExact(TimeSpec ts,TimeSpec ts2)93 bool CompareTimeExact(TimeSpec ts, TimeSpec ts2) {
94   return ts2.tv_sec == ts.tv_sec && ts2.tv_nsec == ts.tv_nsec;
95 }
CompareTimeExact(file_time_type ft,TimeSpec ts)96 bool CompareTimeExact(file_time_type ft, TimeSpec ts) {
97   TimeSpec ts2 = {};
98   if (!ConvertToTimeSpec(ts2, ft))
99     return false;
100   return CompareTimeExact(ts, ts2);
101 }
CompareTimeExact(TimeSpec ts,file_time_type ft)102 bool CompareTimeExact(TimeSpec ts, file_time_type ft) {
103   return CompareTimeExact(ft, ts);
104 }
105 
106 struct Times {
107   TimeSpec access, write;
108 };
109 
GetTimes(path const & p)110 Times GetTimes(path const& p) {
111     StatT st;
112     if (::stat(p.c_str(), &st) == -1) {
113         std::error_code ec(errno, std::generic_category());
114 #ifndef TEST_HAS_NO_EXCEPTIONS
115         throw ec;
116 #else
117         std::cerr << ec.message() << std::endl;
118         std::exit(EXIT_FAILURE);
119 #endif
120     }
121     return {extract_atime(st), extract_mtime(st)};
122 }
123 
LastAccessTime(path const & p)124 TimeSpec LastAccessTime(path const& p) { return GetTimes(p).access; }
125 
LastWriteTime(path const & p)126 TimeSpec LastWriteTime(path const& p) { return GetTimes(p).write; }
127 
GetSymlinkTimes(path const & p)128 Times GetSymlinkTimes(path const& p) {
129   StatT st;
130   if (::lstat(p.c_str(), &st) == -1) {
131     std::error_code ec(errno, std::generic_category());
132 #ifndef TEST_HAS_NO_EXCEPTIONS
133         throw ec;
134 #else
135         std::cerr << ec.message() << std::endl;
136         std::exit(EXIT_FAILURE);
137 #endif
138     }
139     Times res;
140     res.access = extract_atime(st);
141     res.write = extract_mtime(st);
142     return res;
143 }
144 
145 namespace {
146 
147 // In some configurations, the comparison is tautological and the test is valid.
148 // We disable the warning so that we can actually test it regardless. Also, that
149 // diagnostic is pretty new, so also don't fail if old clang does not support it
150 #if defined(__clang__)
151 #pragma clang diagnostic push
152 #pragma clang diagnostic ignored "-Wunknown-warning-option"
153 #pragma clang diagnostic ignored "-Wunknown-pragmas"
154 #pragma clang diagnostic ignored "-Wtautological-constant-compare"
155 #endif
156 
__anon969760ab0202null157 static const bool SupportsNegativeTimes = [] {
158   using namespace std::chrono;
159   std::error_code ec;
160   TimeSpec old_write_time, new_write_time;
161   { // WARNING: Do not assert in this scope.
162     scoped_test_env env;
163     const path file = env.create_file("file", 42);
164     old_write_time = LastWriteTime(file);
165     file_time_type tp(seconds(-5));
166     fs::last_write_time(file, tp, ec);
167     new_write_time = LastWriteTime(file);
168   }
169 
170   return !ec && new_write_time.tv_sec < 0;
171 }();
172 
__anon969760ab0302null173 static const bool SupportsMaxTime = [] {
174   using namespace std::chrono;
175   TimeSpec max_ts = {};
176   if (!ConvertToTimeSpec(max_ts, file_time_type::max()))
177     return false;
178 
179   std::error_code ec;
180   TimeSpec old_write_time, new_write_time;
181   { // WARNING: Do not assert in this scope.
182     scoped_test_env env;
183     const path file = env.create_file("file", 42);
184     old_write_time = LastWriteTime(file);
185     file_time_type tp = file_time_type::max();
186     fs::last_write_time(file, tp, ec);
187     new_write_time = LastWriteTime(file);
188   }
189   return !ec && new_write_time.tv_sec > max_ts.tv_sec - 1;
190 }();
191 
__anon969760ab0402null192 static const bool SupportsMinTime = [] {
193   using namespace std::chrono;
194   TimeSpec min_ts = {};
195   if (!ConvertToTimeSpec(min_ts, file_time_type::min()))
196     return false;
197   std::error_code ec;
198   TimeSpec old_write_time, new_write_time;
199   { // WARNING: Do not assert in this scope.
200     scoped_test_env env;
201     const path file = env.create_file("file", 42);
202     old_write_time = LastWriteTime(file);
203     file_time_type tp = file_time_type::min();
204     fs::last_write_time(file, tp, ec);
205     new_write_time = LastWriteTime(file);
206   }
207   return !ec && new_write_time.tv_sec < min_ts.tv_sec + 1;
208 }();
209 
__anon969760ab0502null210 static const bool SupportsNanosecondRoundTrip = [] {
211   NanoSec ns(3);
212   static_assert(std::is_same<file_time_type::period, std::nano>::value, "");
213 
214   // Test that the system call we use to set the times also supports nanosecond
215   // resolution. (utimes does not)
216   file_time_type ft(ns);
217   {
218     scoped_test_env env;
219     const path p = env.create_file("file", 42);
220     last_write_time(p, ft);
221     return last_write_time(p) == ft;
222   }
223 }();
224 
225 // The HFS+ filesystem (used by default before macOS 10.13) stores timestamps at
226 // a 1-second granularity, and APFS (now the default) at a 1 nanosecond granularity.
227 // 1-second granularity is also the norm on many of the supported filesystems
228 // on Linux as well.
__anon969760ab0602null229 static const bool WorkaroundStatTruncatesToSeconds = [] {
230   MicroSec micros(3);
231   static_assert(std::is_same<file_time_type::period, std::nano>::value, "");
232 
233   file_time_type ft(micros);
234   {
235     scoped_test_env env;
236     const path p = env.create_file("file", 42);
237     if (LastWriteTime(p).tv_nsec != 0)
238       return false;
239     last_write_time(p, ft);
240     return last_write_time(p) != ft && LastWriteTime(p).tv_nsec == 0;
241   }
242 }();
243 
__anon969760ab0702null244 static const bool SupportsMinRoundTrip = [] {
245   TimeSpec ts = {};
246   if (!ConvertToTimeSpec(ts, file_time_type::min()))
247     return false;
248   file_time_type min_val = {};
249   if (!ConvertFromTimeSpec(min_val, ts))
250     return false;
251   return min_val == file_time_type::min();
252 }();
253 
254 } // end namespace
255 
CompareTime(TimeSpec t1,TimeSpec t2)256 static bool CompareTime(TimeSpec t1, TimeSpec t2) {
257   if (SupportsNanosecondRoundTrip)
258     return CompareTimeExact(t1, t2);
259   if (t1.tv_sec != t2.tv_sec)
260     return false;
261 
262   auto diff = std::abs(t1.tv_nsec - t2.tv_nsec);
263   if (WorkaroundStatTruncatesToSeconds)
264    return diff < duration_cast<NanoSec>(Sec(1)).count();
265   return diff < duration_cast<NanoSec>(MicroSec(1)).count();
266 }
267 
CompareTime(file_time_type t1,TimeSpec t2)268 static bool CompareTime(file_time_type t1, TimeSpec t2) {
269   TimeSpec ts1 = {};
270   if (!ConvertToTimeSpec(ts1, t1))
271     return false;
272   return CompareTime(ts1, t2);
273 }
274 
CompareTime(TimeSpec t1,file_time_type t2)275 static bool CompareTime(TimeSpec t1, file_time_type t2) {
276   return CompareTime(t2, t1);
277 }
278 
CompareTime(file_time_type t1,file_time_type t2)279 static bool CompareTime(file_time_type t1, file_time_type t2) {
280   auto min_secs = duration_cast<Sec>(file_time_type::min().time_since_epoch());
281   bool IsMin =
282       t1.time_since_epoch() < min_secs || t2.time_since_epoch() < min_secs;
283 
284   if (SupportsNanosecondRoundTrip && (!IsMin || SupportsMinRoundTrip))
285     return t1 == t2;
286   if (IsMin) {
287     return duration_cast<Sec>(t1.time_since_epoch()) ==
288            duration_cast<Sec>(t2.time_since_epoch());
289   }
290   file_time_type::duration dur;
291   if (t1 > t2)
292     dur = t1 - t2;
293   else
294     dur = t2 - t1;
295   if (WorkaroundStatTruncatesToSeconds)
296     return duration_cast<Sec>(dur).count() == 0;
297   return duration_cast<MicroSec>(dur).count() == 0;
298 }
299 
300 // Check if a time point is representable on a given filesystem. Check that:
301 // (A) 'tp' is representable as a time_t
302 // (B) 'tp' is non-negative or the filesystem supports negative times.
303 // (C) 'tp' is not 'file_time_type::max()' or the filesystem supports the max
304 //     value.
305 // (D) 'tp' is not 'file_time_type::min()' or the filesystem supports the min
306 //     value.
TimeIsRepresentableByFilesystem(file_time_type tp)307 inline bool TimeIsRepresentableByFilesystem(file_time_type tp) {
308   TimeSpec ts = {};
309   if (!ConvertToTimeSpec(ts, tp))
310     return false;
311   else if (tp.time_since_epoch().count() < 0 && !SupportsNegativeTimes)
312     return false;
313   else if (tp == file_time_type::max() && !SupportsMaxTime)
314     return false;
315   else if (tp == file_time_type::min() && !SupportsMinTime)
316     return false;
317   return true;
318 }
319 
320 #if defined(__clang__)
321 #pragma clang diagnostic pop
322 #endif
323 
324 // Create a sub-second duration using the smallest period the filesystem supports.
SubSec(long long val)325 file_time_type::duration SubSec(long long val) {
326   using SubSecT = file_time_type::duration;
327   if (SupportsNanosecondRoundTrip) {
328     return duration_cast<SubSecT>(NanoSec(val));
329   } else {
330     return duration_cast<SubSecT>(MicroSec(val));
331   }
332 }
333 
334 TEST_SUITE(last_write_time_test_suite)
335 
TEST_CASE(signature_test)336 TEST_CASE(signature_test)
337 {
338     const file_time_type t;
339     const path p; ((void)p);
340     std::error_code ec; ((void)ec);
341     ASSERT_SAME_TYPE(decltype(last_write_time(p)), file_time_type);
342     ASSERT_SAME_TYPE(decltype(last_write_time(p, ec)), file_time_type);
343     ASSERT_SAME_TYPE(decltype(last_write_time(p, t)), void);
344     ASSERT_SAME_TYPE(decltype(last_write_time(p, t, ec)), void);
345     ASSERT_NOT_NOEXCEPT(last_write_time(p));
346     ASSERT_NOT_NOEXCEPT(last_write_time(p, t));
347     ASSERT_NOEXCEPT(last_write_time(p, ec));
348     ASSERT_NOEXCEPT(last_write_time(p, t, ec));
349 }
350 
TEST_CASE(read_last_write_time_static_env_test)351 TEST_CASE(read_last_write_time_static_env_test)
352 {
353     using C = file_time_type::clock;
354     file_time_type min = file_time_type::min();
355     {
356         file_time_type ret = last_write_time(StaticEnv::File);
357         TEST_CHECK(ret != min);
358         TEST_CHECK(ret < C::now());
359         TEST_CHECK(CompareTime(ret, LastWriteTime(StaticEnv::File)));
360 
361         file_time_type ret2 = last_write_time(StaticEnv::SymlinkToFile);
362         TEST_CHECK(CompareTime(ret, ret2));
363         TEST_CHECK(CompareTime(ret2, LastWriteTime(StaticEnv::SymlinkToFile)));
364     }
365     {
366         file_time_type ret = last_write_time(StaticEnv::Dir);
367         TEST_CHECK(ret != min);
368         TEST_CHECK(ret < C::now());
369         TEST_CHECK(CompareTime(ret, LastWriteTime(StaticEnv::Dir)));
370 
371         file_time_type ret2 = last_write_time(StaticEnv::SymlinkToDir);
372         TEST_CHECK(CompareTime(ret, ret2));
373         TEST_CHECK(CompareTime(ret2, LastWriteTime(StaticEnv::SymlinkToDir)));
374     }
375 }
376 
TEST_CASE(get_last_write_time_dynamic_env_test)377 TEST_CASE(get_last_write_time_dynamic_env_test)
378 {
379     using Clock = file_time_type::clock;
380     using Sec = std::chrono::seconds;
381     scoped_test_env env;
382 
383     const path file = env.create_file("file", 42);
384     const path dir = env.create_dir("dir");
385 
386     const auto file_times = GetTimes(file);
387     const TimeSpec file_write_time = file_times.write;
388     const auto dir_times = GetTimes(dir);
389     const TimeSpec dir_write_time = dir_times.write;
390 
391     file_time_type ftime = last_write_time(file);
392     TEST_CHECK(Clock::to_time_t(ftime) == file_write_time.tv_sec);
393     TEST_CHECK(CompareTime(ftime, file_write_time));
394 
395     file_time_type dtime = last_write_time(dir);
396     TEST_CHECK(Clock::to_time_t(dtime) == dir_write_time.tv_sec);
397     TEST_CHECK(CompareTime(dtime, dir_write_time));
398 
399     SleepFor(Sec(2));
400 
401     // update file and add a file to the directory. Make sure the times increase.
402     std::ofstream of(file, std::ofstream::app);
403     of << "hello";
404     of.close();
405     env.create_file("dir/file1", 1);
406 
407     file_time_type ftime2 = last_write_time(file);
408     file_time_type dtime2 = last_write_time(dir);
409 
410     TEST_CHECK(ftime2 > ftime);
411     TEST_CHECK(dtime2 > dtime);
412     TEST_CHECK(CompareTime(LastWriteTime(file), ftime2));
413     TEST_CHECK(CompareTime(LastWriteTime(dir), dtime2));
414 }
415 
416 
TEST_CASE(set_last_write_time_dynamic_env_test)417 TEST_CASE(set_last_write_time_dynamic_env_test)
418 {
419     using Clock = file_time_type::clock;
420     scoped_test_env env;
421 
422     const path file = env.create_file("file", 42);
423     const path dir = env.create_dir("dir");
424     const auto now = Clock::now();
425     const file_time_type epoch_time = now - now.time_since_epoch();
426 
427     const file_time_type future_time = now + Hours(3) + Sec(42) + SubSec(17);
428     const file_time_type past_time = now - Minutes(3) - Sec(42) - SubSec(17);
429     const file_time_type before_epoch_time =
430         epoch_time - Minutes(3) - Sec(42) - SubSec(17);
431     // FreeBSD has a bug in their utimes implementation where the time is not update
432     // when the number of seconds is '-1'.
433 #if defined(__FreeBSD__) || defined(__NetBSD__)
434     const file_time_type just_before_epoch_time =
435         epoch_time - Sec(2) - SubSec(17);
436 #else
437     const file_time_type just_before_epoch_time = epoch_time - SubSec(17);
438 #endif
439 
440     struct TestCase {
441       const char * case_name;
442       path p;
443       file_time_type new_time;
444     } cases[] = {
445         {"file, epoch_time", file, epoch_time},
446         {"dir, epoch_time", dir, epoch_time},
447         {"file, future_time", file, future_time},
448         {"dir, future_time", dir, future_time},
449         {"file, past_time", file, past_time},
450         {"dir, past_time", dir, past_time},
451         {"file, before_epoch_time", file, before_epoch_time},
452         {"dir, before_epoch_time", dir, before_epoch_time},
453         {"file, just_before_epoch_time", file, just_before_epoch_time},
454         {"dir, just_before_epoch_time", dir, just_before_epoch_time}
455     };
456     for (const auto& TC : cases) {
457         std::cerr << "Test Case = " << TC.case_name << "\n";
458         const auto old_times = GetTimes(TC.p);
459         file_time_type old_time;
460         TEST_REQUIRE(ConvertFromTimeSpec(old_time, old_times.write));
461 
462         std::error_code ec = GetTestEC();
463         last_write_time(TC.p, TC.new_time, ec);
464         TEST_CHECK(!ec);
465 
466         ec = GetTestEC();
467         file_time_type  got_time = last_write_time(TC.p, ec);
468         TEST_REQUIRE(!ec);
469 
470         if (TimeIsRepresentableByFilesystem(TC.new_time)) {
471             TEST_CHECK(got_time != old_time);
472             TEST_CHECK(CompareTime(got_time, TC.new_time));
473             TEST_CHECK(CompareTime(LastAccessTime(TC.p), old_times.access));
474         }
475     }
476 }
477 
TEST_CASE(last_write_time_symlink_test)478 TEST_CASE(last_write_time_symlink_test)
479 {
480     using Clock = file_time_type::clock;
481 
482     scoped_test_env env;
483 
484     const path file = env.create_file("file", 42);
485     const path sym = env.create_symlink("file", "sym");
486 
487     const file_time_type new_time = Clock::now() + Hours(3);
488 
489     const auto old_times = GetTimes(sym);
490     const auto old_sym_times = GetSymlinkTimes(sym);
491 
492     std::error_code ec = GetTestEC();
493     last_write_time(sym, new_time, ec);
494     TEST_CHECK(!ec);
495 
496     file_time_type  got_time = last_write_time(sym);
497     TEST_CHECK(!CompareTime(got_time, old_times.write));
498     if (!WorkaroundStatTruncatesToSeconds) {
499       TEST_CHECK(got_time == new_time);
500     } else {
501       TEST_CHECK(CompareTime(got_time, new_time));
502     }
503 
504     TEST_CHECK(CompareTime(LastWriteTime(file), new_time));
505     TEST_CHECK(CompareTime(LastAccessTime(sym), old_times.access));
506     Times sym_times = GetSymlinkTimes(sym);
507     TEST_CHECK(CompareTime(sym_times.write, old_sym_times.write));
508 }
509 
510 
TEST_CASE(test_write_min_time)511 TEST_CASE(test_write_min_time)
512 {
513     scoped_test_env env;
514     const path p = env.create_file("file", 42);
515     const file_time_type old_time = last_write_time(p);
516     file_time_type new_time = file_time_type::min();
517 
518     std::error_code ec = GetTestEC();
519     last_write_time(p, new_time, ec);
520     file_time_type tt = last_write_time(p);
521 
522     if (TimeIsRepresentableByFilesystem(new_time)) {
523         TEST_CHECK(!ec);
524         TEST_CHECK(CompareTime(tt, new_time));
525 
526         last_write_time(p, old_time);
527         new_time = file_time_type::min() + SubSec(1);
528 
529         ec = GetTestEC();
530         last_write_time(p, new_time, ec);
531         tt = last_write_time(p);
532 
533         if (TimeIsRepresentableByFilesystem(new_time)) {
534             TEST_CHECK(!ec);
535             TEST_CHECK(CompareTime(tt, new_time));
536         } else {
537           TEST_CHECK(ErrorIs(ec, std::errc::value_too_large));
538           TEST_CHECK(tt == old_time);
539         }
540     } else {
541       TEST_CHECK(ErrorIs(ec, std::errc::value_too_large));
542       TEST_CHECK(tt == old_time);
543     }
544 }
545 
TEST_CASE(test_write_max_time)546 TEST_CASE(test_write_max_time) {
547   scoped_test_env env;
548   const path p = env.create_file("file", 42);
549   const file_time_type old_time = last_write_time(p);
550   file_time_type new_time = file_time_type::max();
551 
552   std::error_code ec = GetTestEC();
553   last_write_time(p, new_time, ec);
554   file_time_type tt = last_write_time(p);
555 
556   if (TimeIsRepresentableByFilesystem(new_time)) {
557     TEST_CHECK(!ec);
558     TEST_CHECK(CompareTime(tt, new_time));
559   } else {
560     TEST_CHECK(ErrorIs(ec, std::errc::value_too_large));
561     TEST_CHECK(tt == old_time);
562   }
563 }
564 
TEST_CASE(test_value_on_failure)565 TEST_CASE(test_value_on_failure)
566 {
567     const path p = StaticEnv::DNE;
568     std::error_code ec = GetTestEC();
569     TEST_CHECK(last_write_time(p, ec) == file_time_type::min());
570     TEST_CHECK(ErrorIs(ec, std::errc::no_such_file_or_directory));
571 }
572 
TEST_CASE(test_exists_fails)573 TEST_CASE(test_exists_fails)
574 {
575     scoped_test_env env;
576     const path dir = env.create_dir("dir");
577     const path file = env.create_file("dir/file", 42);
578     permissions(dir, perms::none);
579 
580     std::error_code ec = GetTestEC();
581     TEST_CHECK(last_write_time(file, ec) == file_time_type::min());
582     TEST_CHECK(ErrorIs(ec, std::errc::permission_denied));
583 
584     ExceptionChecker Checker(file, std::errc::permission_denied,
585                              "last_write_time");
586     TEST_CHECK_THROW_RESULT(filesystem_error, Checker, last_write_time(file));
587 }
588 
589 TEST_SUITE_END()
590