// Copyright (C) 2003-2018 Free Software Foundation, Inc. // // This file is part of the GNU ISO C++ Library. This library is free // software; you can redistribute it and/or modify it under the // terms of the GNU General Public License as published by the // Free Software Foundation; either version 3, or (at your option) // any later version. // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // You should have received a copy of the GNU General Public License along // with this library; see the file COPYING3. If not see // . // 27.8.1.4 Overridden virtual functions #include #include #include #include template class checksumcvt : public std::codecvt { typedef std::codecvt Base; static const size_t width = sizeof(InternT) + 1; public: typedef InternT intern_type; typedef char extern_type; explicit checksumcvt(size_t refs = 0) : Base(refs) { } protected: virtual typename std::codecvt::result do_out(StateT&, const intern_type* from, const intern_type* from_end, const intern_type*& from_next, extern_type* to, extern_type* to_end, extern_type*& to_next) const { size_t len = std::min(static_cast(from_end - from), static_cast(to_end - to) / width); while (len--) { const char* p = reinterpret_cast(from); unsigned char checksum = 0; for (size_t i = 0; i < sizeof(intern_type); ++i) { *to++ = p[i]; checksum ^= static_cast(p[i]); } *to++ = checksum; ++from; } from_next = from; to_next = to; return from_next == from_end ? std::codecvt::ok : std::codecvt::partial; } virtual typename std::codecvt::result do_unshift(StateT&, extern_type* to, extern_type*, extern_type*& to_next) const { to_next = to; return std::codecvt::ok; } virtual typename std::codecvt::result do_in(StateT&, const extern_type* from, const extern_type* from_end, const extern_type*& from_next, intern_type* to, intern_type* to_end, intern_type*& to_next) const { size_t len = std::min(static_cast(to_end - to), static_cast(from_end - from) / width); while (len) { const char* f = from; intern_type tmp; char* p = reinterpret_cast(&tmp); unsigned char checksum = 0; for (size_t i = 0; i < sizeof(intern_type); ++i) { p[i] = *f; checksum ^= static_cast(*f++); } if (*f++ != checksum) break; from = f; *to++ = tmp; len--; } from_next = from; to_next = to; return len ? std::codecvt::error : (from_next == from_end ? std::codecvt::ok : std::codecvt::partial); } virtual int do_encoding() const throw() { return width; } virtual int do_length(StateT&, const extern_type* from, const extern_type* end, size_t max) const { size_t len = std::min(max, static_cast(end - from) / width); int ret = 0; while (len--) { unsigned char checksum = 0; for (size_t i = 0; i < sizeof(intern_type); ++i) { checksum ^= static_cast(*from++); } if (*from++ != checksum) break; ret++; } return ret; } virtual int do_max_length() const throw() { return width; } virtual bool do_always_noconv() const throw() { return false; } }; // libstdc++/11544 (invalid byte sequence in file) void test02() { using namespace std; locale loc(locale::classic(), new checksumcvt); const char* name = "tmp_11544-2"; FILE* f = fopen(name, "w"); VERIFY( fwrite("aaaab", 1, 5, f) == 5 ); fclose(f); wifstream in; in.imbue(loc); in.open(name); VERIFY( in.good() ); in.get(); VERIFY( !in.good() ); VERIFY( in.bad() ); VERIFY( !in.eof() ); in.close(); } int main() { test02(); return 0; }