mirror of
https://github.com/shadps4-emu/ext-fmt.git
synced 2024-11-27 03:30:32 +00:00
Implement locking
This commit is contained in:
parent
c5a85f8d7d
commit
e450b7aeb3
@ -144,4 +144,33 @@ TEST(scan_test, file) {
|
|||||||
EXPECT_EQ(n1, 10);
|
EXPECT_EQ(n1, 10);
|
||||||
EXPECT_EQ(n2, 20);
|
EXPECT_EQ(n2, 20);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(scan_test, lock) {
|
||||||
|
fmt::file read_end, write_end;
|
||||||
|
fmt::file::pipe(read_end, write_end);
|
||||||
|
|
||||||
|
std::thread producer([&]() {
|
||||||
|
fmt::string_view input = "42 ";
|
||||||
|
for (int i = 0; i < 1000; ++i)
|
||||||
|
write_end.write(input.data(), input.size());
|
||||||
|
write_end.close();
|
||||||
|
});
|
||||||
|
|
||||||
|
fmt::buffered_file f = read_end.fdopen("r");
|
||||||
|
auto fun = [&]() {
|
||||||
|
int value = 0;
|
||||||
|
while (fmt::scan(f.get(), "{}", value)) {
|
||||||
|
if (value != 42) {
|
||||||
|
read_end.close();
|
||||||
|
EXPECT_EQ(value, 42);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
std::thread consumer1(fun);
|
||||||
|
std::thread consumer2(fun);
|
||||||
|
producer.join();
|
||||||
|
consumer1.join();
|
||||||
|
consumer2.join();
|
||||||
|
}
|
||||||
#endif // FMT_USE_FCNTL
|
#endif // FMT_USE_FCNTL
|
||||||
|
22
test/scan.h
22
test/scan.h
@ -266,9 +266,20 @@ class file_scan_buffer : public scan_buffer {
|
|||||||
public:
|
public:
|
||||||
explicit file_scan_buffer(FILE* f)
|
explicit file_scan_buffer(FILE* f)
|
||||||
: scan_buffer(nullptr, nullptr, false), file_(f) {
|
: scan_buffer(nullptr, nullptr, false), file_(f) {
|
||||||
// TODO: lock file?
|
#ifndef _WIN32
|
||||||
|
flockfile(f);
|
||||||
|
#else
|
||||||
|
_lock_file(f);
|
||||||
|
#endif
|
||||||
fill();
|
fill();
|
||||||
}
|
}
|
||||||
|
~file_scan_buffer() {
|
||||||
|
#ifndef _WIN32
|
||||||
|
funlockfile(file_);
|
||||||
|
#else
|
||||||
|
_unlock_file(file_);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
};
|
};
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
@ -472,7 +483,6 @@ struct scan_handler : error_handler {
|
|||||||
switch (arg_.type) {
|
switch (arg_.type) {
|
||||||
case scan_type::int_type:
|
case scan_type::int_type:
|
||||||
if (auto value = read_int()) *arg_.int_value = *value;
|
if (auto value = read_int()) *arg_.int_value = *value;
|
||||||
// TODO: stop on end of input
|
|
||||||
break;
|
break;
|
||||||
case scan_type::uint_type:
|
case scan_type::uint_type:
|
||||||
if (auto value = read_uint()) *arg_.uint_value = *value;
|
if (auto value = read_uint()) *arg_.uint_value = *value;
|
||||||
@ -512,6 +522,11 @@ struct scan_handler : error_handler {
|
|||||||
arg_.custom.scan(arg_.custom.value, parse_ctx_, scan_ctx_);
|
arg_.custom.scan(arg_.custom.value, parse_ctx_, scan_ctx_);
|
||||||
return parse_ctx_.begin();
|
return parse_ctx_.begin();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void on_error(const char* message) {
|
||||||
|
scan_ctx_.advance_to(scan_ctx_.end());
|
||||||
|
error_handler::on_error(message);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
} // namespace detail
|
} // namespace detail
|
||||||
|
|
||||||
@ -533,9 +548,10 @@ auto scan(string_view input, string_view fmt, T&... args)
|
|||||||
return input.begin() + (buf.begin().base() - input.data());
|
return input.begin() + (buf.begin().base() - input.data());
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename... T> void scan(std::FILE* f, string_view fmt, T&... args) {
|
template <typename... T> bool scan(std::FILE* f, string_view fmt, T&... args) {
|
||||||
auto&& buf = detail::file_scan_buffer(f);
|
auto&& buf = detail::file_scan_buffer(f);
|
||||||
vscan(buf, fmt, make_scan_args(args...));
|
vscan(buf, fmt, make_scan_args(args...));
|
||||||
|
return buf.begin() != buf.end();
|
||||||
}
|
}
|
||||||
|
|
||||||
FMT_END_NAMESPACE
|
FMT_END_NAMESPACE
|
||||||
|
Loading…
Reference in New Issue
Block a user