This fixes a crash that happens in std::filebuf when a large read consumes the entire get area and is followed by a write, which is then synced to the file by a call to overflow.
The problem is that xsgetn calls _M_set_buffer(0) after reading from the file (i.e. when in 'read' mode). As the comments on _M_set_buffer say, an argument of 0 is used for 'write' mode. This causes the filebuf to have an active put area while in 'read' mode, so that the next write inserts straight into that put area, rather than performing the required seek to leave 'read' mode. The next overflow then tries to leave 'read' mode by doing a seek, but that then tries to flush the non-empty put area by calling overflow, which goes into a loop until we overflow the stack. The solution is to simply remove the call to _M_set_buffer(0). It's not needed because the buffers are already set up appropriately after xsgetn has read from the file: there's no active putback, no put area, and setg(eback(), egptr(), egptr()) has been called so there's nothing available in the get area. All we need to do is set _M_reading = true so that a following write knows it needs to perform a seek. The new testcase passes with GCC 4.5, so this is technically a regression. However, I have a more demanding test that fails even with GCC 4.5, so I don't think mixing reads and writes without intervening seeks was ever working completely. I hope it is now. I spent a LOT of time checking the make check-performance results before and after this patch (and with various other attempted fixes) and any difference seemed to be noise. PR libstdc++/81395 * include/bits/fstream.tcc (basic_filebuf::xsgetn): Don't set buffer pointers for write mode after reading. * testsuite/27_io/basic_filebuf/sgetn/char/81395.cc: New. Tested powerpc64le-linux, committed to trunk.
commit 535a7ea29b4d6724519c0f472bcfe3eb9d79070a Author: Jonathan Wakely <jwak...@redhat.com> Date: Tue Jul 18 15:20:25 2017 +0100 PR libstdc++/81395 fix crash when write follows large read PR libstdc++/81395 * include/bits/fstream.tcc (basic_filebuf::xsgetn): Don't set buffer pointers for write mode after reading. * testsuite/27_io/basic_filebuf/sgetn/char/81395.cc: New. diff --git a/libstdc++-v3/include/bits/fstream.tcc b/libstdc++-v3/include/bits/fstream.tcc index b1beff86..ef51a84 100644 --- a/libstdc++-v3/include/bits/fstream.tcc +++ b/libstdc++-v3/include/bits/fstream.tcc @@ -699,7 +699,7 @@ _GLIBCXX_BEGIN_NAMESPACE_VERSION if (__n == 0) { - _M_set_buffer(0); + // Set _M_reading. Buffer is already in initial 'read' mode. _M_reading = true; } else if (__len == 0) diff --git a/libstdc++-v3/testsuite/27_io/basic_filebuf/sgetn/char/81395.cc b/libstdc++-v3/testsuite/27_io/basic_filebuf/sgetn/char/81395.cc new file mode 100644 index 0000000..4985628 --- /dev/null +++ b/libstdc++-v3/testsuite/27_io/basic_filebuf/sgetn/char/81395.cc @@ -0,0 +1,46 @@ +// Copyright (C) 2017 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 +// <http://www.gnu.org/licenses/>. + +// { dg-do run } + +// PR libstdc++/81395 + +#include <fstream> +#include <cstring> // for std::memset +#include <cstdio> // For BUFSIZ + +using std::memset; + +int main() +{ + { + std::filebuf fb; + fb.open("test.txt", std::ios::out); + char data[BUFSIZ]; + memset(data, 'A', sizeof(data)); + fb.sputn(data, sizeof(data)); + } + + std::filebuf fb; + fb.open("test.txt", std::ios::in|std::ios::out); + char buf[BUFSIZ]; + memset(buf, 0, sizeof(buf)); + fb.sgetn(buf, sizeof(buf)); + // Switch from reading to writing without seeking first: + fb.sputn("B", 1); + fb.pubsync(); +}