mirror of
https://github.com/pybind/pybind11.git
synced 2024-11-22 05:05:11 +00:00
pythonbuf fix (#2675)
* Added test_thread testing for ostream_redirect segfault recreation * fix: scoped_ostream_redirect str created outside gil * Moved threading tests into test_iostream. Cleaned up some formatting. Deleted test_thread.{cpp,py} * CI: few formatting fixes * CI: yet another formatting fix * CI: more formatting fixes. Removed unecessary comment * Ignore 'warning C4702: unreachable code' in MSVC 2015 Co-authored-by: Nick Bridge <nick.bridge.chess@gmail.com> Co-authored-by: Nick Bridge <nbridge@jumptrading.com> Co-authored-by: Yannick Jadoul <yannick.jadoul@belgacom.net>
This commit is contained in:
parent
17c22b9e0f
commit
2fa4747cd4
@ -43,11 +43,13 @@ private:
|
|||||||
// simplified to a fully qualified call.
|
// simplified to a fully qualified call.
|
||||||
int _sync() {
|
int _sync() {
|
||||||
if (pbase() != pptr()) {
|
if (pbase() != pptr()) {
|
||||||
// This subtraction cannot be negative, so dropping the sign
|
|
||||||
str line(pbase(), static_cast<size_t>(pptr() - pbase()));
|
|
||||||
|
|
||||||
{
|
{
|
||||||
gil_scoped_acquire tmp;
|
gil_scoped_acquire tmp;
|
||||||
|
|
||||||
|
// This subtraction cannot be negative, so dropping the sign.
|
||||||
|
str line(pbase(), static_cast<size_t>(pptr() - pbase()));
|
||||||
|
|
||||||
pywrite(line);
|
pywrite(line);
|
||||||
pyflush();
|
pyflush();
|
||||||
}
|
}
|
||||||
|
@ -7,10 +7,15 @@
|
|||||||
BSD-style license that can be found in the LICENSE file.
|
BSD-style license that can be found in the LICENSE file.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#if defined(_MSC_VER) && _MSC_VER < 1910 // VS 2015's MSVC
|
||||||
|
# pragma warning(disable: 4702) // unreachable code in system header (xatomic.h(382))
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <pybind11/iostream.h>
|
#include <pybind11/iostream.h>
|
||||||
#include "pybind11_tests.h"
|
#include "pybind11_tests.h"
|
||||||
|
#include <atomic>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
#include <thread>
|
||||||
|
|
||||||
|
|
||||||
void noisy_function(std::string msg, bool flush) {
|
void noisy_function(std::string msg, bool flush) {
|
||||||
@ -25,6 +30,40 @@ void noisy_funct_dual(std::string msg, std::string emsg) {
|
|||||||
std::cerr << emsg;
|
std::cerr << emsg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// object to manage C++ thread
|
||||||
|
// simply repeatedly write to std::cerr until stopped
|
||||||
|
// redirect is called at some point to test the safety of scoped_estream_redirect
|
||||||
|
struct TestThread {
|
||||||
|
TestThread() : t_{nullptr}, stop_{false} {
|
||||||
|
auto thread_f = [this] {
|
||||||
|
while (!stop_) {
|
||||||
|
std::cout << "x" << std::flush;
|
||||||
|
std::this_thread::sleep_for(std::chrono::microseconds(50));
|
||||||
|
} };
|
||||||
|
t_ = new std::thread(std::move(thread_f));
|
||||||
|
}
|
||||||
|
|
||||||
|
~TestThread() {
|
||||||
|
delete t_;
|
||||||
|
}
|
||||||
|
|
||||||
|
void stop() { stop_ = true; }
|
||||||
|
|
||||||
|
void join() {
|
||||||
|
py::gil_scoped_release gil_lock;
|
||||||
|
t_->join();
|
||||||
|
}
|
||||||
|
|
||||||
|
void sleep() {
|
||||||
|
py::gil_scoped_release gil_lock;
|
||||||
|
std::this_thread::sleep_for(std::chrono::milliseconds(50));
|
||||||
|
}
|
||||||
|
|
||||||
|
std::thread * t_;
|
||||||
|
std::atomic<bool> stop_;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
TEST_SUBMODULE(iostream, m) {
|
TEST_SUBMODULE(iostream, m) {
|
||||||
|
|
||||||
add_ostream_redirect(m);
|
add_ostream_redirect(m);
|
||||||
@ -70,4 +109,10 @@ TEST_SUBMODULE(iostream, m) {
|
|||||||
std::cout << msg << std::flush;
|
std::cout << msg << std::flush;
|
||||||
std::cerr << emsg << std::flush;
|
std::cerr << emsg << std::flush;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
py::class_<TestThread>(m, "TestThread")
|
||||||
|
.def(py::init<>())
|
||||||
|
.def("stop", &TestThread::stop)
|
||||||
|
.def("join", &TestThread::join)
|
||||||
|
.def("sleep", &TestThread::sleep);
|
||||||
}
|
}
|
||||||
|
@ -216,3 +216,26 @@ def test_redirect_both(capfd):
|
|||||||
assert stderr == ""
|
assert stderr == ""
|
||||||
assert stream.getvalue() == msg
|
assert stream.getvalue() == msg
|
||||||
assert stream2.getvalue() == msg2
|
assert stream2.getvalue() == msg2
|
||||||
|
|
||||||
|
|
||||||
|
def test_threading():
|
||||||
|
with m.ostream_redirect(stdout=True, stderr=False):
|
||||||
|
# start some threads
|
||||||
|
threads = []
|
||||||
|
|
||||||
|
# start some threads
|
||||||
|
for _j in range(20):
|
||||||
|
threads.append(m.TestThread())
|
||||||
|
|
||||||
|
# give the threads some time to fail
|
||||||
|
threads[0].sleep()
|
||||||
|
|
||||||
|
# stop all the threads
|
||||||
|
for t in threads:
|
||||||
|
t.stop()
|
||||||
|
|
||||||
|
for t in threads:
|
||||||
|
t.join()
|
||||||
|
|
||||||
|
# if a thread segfaults, we don't get here
|
||||||
|
assert True
|
||||||
|
@ -8,8 +8,8 @@
|
|||||||
BSD-style license that can be found in the LICENSE file.
|
BSD-style license that can be found in the LICENSE file.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#if defined(_MSC_VER) && _MSC_VER < 1910
|
#if defined(_MSC_VER) && _MSC_VER < 1910 // VS 2015's MSVC
|
||||||
# pragma warning(disable: 4702) // unreachable code in system header
|
# pragma warning(disable: 4702) // unreachable code in system header (xatomic.h(382))
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "pybind11_tests.h"
|
#include "pybind11_tests.h"
|
||||||
|
Loading…
Reference in New Issue
Block a user