Bug #15130
closedNested Timeouts hang on cygwin
Description
After using make; make runnable, the following one-liner hangs:
 ./ruby -ropen-uri -e 'open "http://web.mit.edu/index.html"'
This happens on any http URI, but not for local files. The above URI is just an example. I would have chosen ruby-lang.org if it was still using http, but I didn't want to complicate this with https.
The open work for local files. I discovered this when trying to use make up, where the downloading of Unicode data files just hung.
Here's the trace I get after hitting Ctrl-C:
Traceback (most recent call last):
15: from -e:1:in <main>' 14: from /cygdrive/c/Data/ruby_test_copy/lib/open-uri.rb:35:in open'
13: from /cygdrive/c/Data/ruby_test_copy/lib/open-uri.rb:736:in open' 12: from /cygdrive/c/Data/ruby_test_copy/lib/open-uri.rb:165:in open_uri'
11: from /cygdrive/c/Data/ruby_test_copy/lib/open-uri.rb:224:in open_loop' 10: from /cygdrive/c/Data/ruby_test_copy/lib/open-uri.rb:224:in catch'
9: from /cygdrive/c/Data/ruby_test_copy/lib/open-uri.rb:226:in block in open_loop' 8: from /cygdrive/c/Data/ruby_test_copy/lib/open-uri.rb:758:in buffer_open'
7: from /cygdrive/c/Data/ruby_test_copy/lib/open-uri.rb:337:in open_http' 6: from /cygdrive/c/Data/ruby_test_copy/lib/net/http.rb:924:in start'
5: from /cygdrive/c/Data/ruby_test_copy/lib/net/http.rb:935:in do_start' 4: from /cygdrive/c/Data/ruby_test_copy/lib/net/http.rb:950:in connect'
3: from /cygdrive/c/Data/ruby_test_copy/lib/timeout.rb:103:in timeout' 2: from /cygdrive/c/Data/ruby_test_copy/lib/timeout.rb:97:in block in timeout'
1: from /cygdrive/c/Data/ruby_test_copy/lib/timeout.rb:97:in ensure in block in timeout' /cygdrive/c/Data/ruby_test_copy/lib/timeout.rb:97:in join': Interrupt
My plan is to (manually) bisect this to find the offending commit. One suspicion I have is that the problem is related to timeouts, but this is not confirmed at all. Any help is appreciated.
Files
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      - Related to Bug #15072: thread.c:4356:5: error: implicit declaration of function ‘ubf_list_atfork’ added
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      This issue doesn't exist before #15072 started at r64485 and made it impossible to check whether open-uri works or not. I'm assigning this to Eric (normalperson) because it may well be timeout-related. Sorry if that turns out not to be the case.
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      - Assignee set to normalperson (Eric Wong)
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      I think I know what's going on (but need to do offline stuff).
Should have a patch for you to test in a few hours.
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      - File 0001-thread_pthread.c-fall-back-to-UBF_TIMER_PTHREAD-for-.patch 0001-thread_pthread.c-fall-back-to-UBF_TIMER_PTHREAD-for-.patch added
Martin: can you try the attached patch? And please commit it if it works.
    thread_pthread.c: fall back to UBF_TIMER_PTHREAD for Cygwin
    UBF_TIMER_NONE is not useful until we can make all I/O
    non-blocking <https://bugs.ruby-lang.org/issues/14968>
    For now, UBF_TIMER_PTHREAD should be a safe bet even for
    systems (e.g. Cygwin) without ubf_list.
    [Bug #15130]
Thanks
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      normalperson (Eric Wong) wrote:
Martin: can you try the attached patch? And please commit it if it works.
Thanks a lot for your patch. I'm working on trying it out, but I will only have the results later in my (Japanese) afternoon, sorry.
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      I applied the patch, but got the same problem: I had to use Ctrl-C, and got essentially the same backtrace:
Traceback (most recent call last):
15: from -e:1:in <main>' 14: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/open-uri.rb:35:in open'
13: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/open-uri.rb:736:in open' 12: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/open-uri.rb:165:in open_uri'
11: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/open-uri.rb:224:in open_loop' 10: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/open-uri.rb:224:in catch'
9: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/open-uri.rb:226:in block in open_loop' 8: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/open-uri.rb:756:in buffer_open'
7: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/open-uri.rb:337:in open_http' 6: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/net/http.rb:922:in start'
5: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/net/http.rb:933:in do_start' 4: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/net/http.rb:948:in connect'
3: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/timeout.rb:103:in timeout' 2: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/timeout.rb:97:in block in timeout'
1: from /cygdrive/c/Data/ruby_test_patch_pthread/lib/timeout.rb:97:in ensure in block in timeout' /cygdrive/c/Data/ruby_test_patch_pthread/lib/timeout.rb:97:in join': Interrupt
Please tell me what I can do to help you solve this problem.
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      duerst@it.aoyama.ac.jp wrote:
Please tell me what I can do to help you solve this problem.
Oops, I guess pthread_kill isn't going to work in cygwin :x
This brings the Cygwin pthreads timer thread closer to what was
in 2.5 by setting ec->interrupt_flag (via rb_threadptr_check_signal):
https://80x24.org/spew/20180923084138.8015-1-e@80x24.org/raw
I suspect the "timer_thread_function()" call is redundant and you
can comment it out, too.
Honestly, if this doesn't work, I don't know how 2.5 and earlier
worked with Cygwin...
Also, does this work if you use other Windows environments which
use thread_win32.c instead?
Thanks for your patience.
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      normalperson (Eric Wong) wrote:
This brings the Cygwin pthreads timer thread closer to what was
in 2.5 by setting ec->interrupt_flag (via rb_threadptr_check_signal):https://80x24.org/spew/20180923084138.8015-1-e@80x24.org/raw
Thanks!
Should I use together with or separately from 0001-thread_pthread.c-fall-back-to-UBF_TIMER_PTHREAD-for-.patch ?
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      What I was wondering is: Why do things get stuck in a case where the Timeout isn't actually invoked (as far as I understand). Shouldn't the program just move ahead? Any background information on what may be happening and why is appreciated; it may help me contribute to the solution.
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      duerst@it.aoyama.ac.jp wrote:
normalperson (Eric Wong) wrote:
This brings the Cygwin pthreads timer thread closer to what was
in 2.5 by setting ec->interrupt_flag (via rb_threadptr_check_signal):https://80x24.org/spew/20180923084138.8015-1-e@80x24.org/raw
Thanks!
Should I use together with or separately from 0001-thread_pthread.c-fall-back-to-UBF_TIMER_PTHREAD-for-.patch ?
Together with previous patch.
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      duerst@it.aoyama.ac.jp wrote:
What I was wondering is: Why do things get stuck in a case
where the Timeout isn't actually invoked (as far as I
understand). Shouldn't the program just move ahead? Any
background information on what may be happening and why is
appreciated; it may help me contribute to the solution.
I'm actually not sure how things worked at all under 2.5 for
Windows with Cygwin; just trying to give you a code path which
resembles the 2.5 code paths.
Perhaps @usa (Usaku NAKAMURA) or another Windows expert knows how interrupts work?
Is there something which shows system call traces which could be
useful in diagnosing this?
/me goes afk again for a while...
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      normalperson (Eric Wong) wrote:
Is there something which shows system call traces which could be
useful in diagnosing this?
strace seems to do that, see https://cygwin.com/cygwin-ug-net/strace.html.
strace ./ruby -ropen-uri -e 'open "http://web.mit.edu/index.html"'
produces something like 34 thousand lines out output.
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      I managed to create a very short script that seem so recreate the problem (or at least something similar).
It seems to happen with a Timeout inside a Timeout (I didn't manage to create a problem with only one Timeout).
Here's the script:
require 'timeout'
def time_out
  begin
    Timeout.timeout(3) do |_|
      puts 'Before sleeping in method'
      sleep 1
      puts 'After sleeping in method'
    end
  rescue Timeout::Error
    puts 'Timeout triggered in method'
  end
end
begin
  Timeout.timeout(5) do |_|
    puts 'Before sleeping'
    sleep 1
    puts 'After sleeping, before method call'
    time_out
    puts 'After method call'
  end
rescue Timeout::Error
  puts 'Timeout triggered'
end
When I run this, I get all the expected output, as follows:
Before sleeping
After sleeping, before method call
Before sleeping in method
After sleeping in method
but this script, rather than stopping after that (2 seconds overall), just hangs (for minutes), until I use Ctrl-C.
At that point, it prints out
Traceback (most recent call last):
../timeout_test.rb: execution expired (Timeout::Error)
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      duerst@it.aoyama.ac.jp wrote:
normalperson (Eric Wong) wrote:
Is there something which shows system call traces which could be
useful in diagnosing this?
straceseems to do that, see https://cygwin.com/cygwin-ug-net/strace.html.
OK, same as on Linux, I guess?  (`-f' to trace children should
be useful.
So, are you also confirming my two patches together do NOT work
for you?  (I somewhat expected them to, since they try to follow
the 2.5 code path...)
strace ./ruby -ropen-uri -e 'open "http://web.mit.edu/index.html"'
produces something like 34 thousand lines out output.
Hint #1: --disable=gems to reduce syscalls
Hint #2: add `warn' statements before and after the parts you're
interested in, which makes it easier to find the part of the
code you want to trace:
e.g.:
warn "BEGIN FOO"
.... # whatever you're interested in
warn "END FOO"
You can the search for "BEGIN FOO"/"END FOO" in the strace
output from write/writev syscalls, and everything in between is
what you care about.
When I debug in C, I may also use write to a bogus FD to make
things easier-to-find:
write(-13, "blah blah\n", 10); errno = 0;
That Cygwin manpage doesn't give strace output examples, so I
also wonder if the output is capable of showing strings (I'm
spoiled by Linux strace, which does).  Sometimes I will use
weird/uncommon syscalls or sequences if string matches don't
work; but same idea above for finding unique/rare strings.
But yeah, I also have no idea what syscalls you'd see in
Windows.  At least on Linux I would start looking for where
socket() is called along with nearby clone() syscalls for
Timeout...
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      duerst@it.aoyama.ac.jp wrote:
When I run this, I get all the expected output, as follows:
Before sleeping After sleeping, before method call Before sleeping in method After sleeping in methodbut this script, rather than stopping after that (2 seconds overall), just hangs (for minutes), until I use Ctrl-C.
If you don't hit Ctrl-C, what is the CPU usage of that script?
(from top(1) or similar tools)
Also, that would be a good time to attach "strace -f -p "
and see what is going on.  Thanks
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      normalperson (Eric Wong) wrote:
Eric, many thanks for your answers, and sorry to be late with my replies.
If you don't hit Ctrl-C, what is the CPU usage of that script?
(from top(1) or similar tools)
Using the Windows Task Manager, it's 0.0%.
Also, that would be a good time to attach "strace -f -p "
and see what is going on. Thanks
Here is what I get:
$ strace -f -p 21772
--- Process 14556 created
--- Process 14556 loaded C:\Windows\System32\ntdll.dll at 00007ffaccec0000
--- Process 14556 thread 20116 created
--- Process 14556 thread 14344 created
--- Process 14556 thread 10272 created
--- Process 14556 loaded C:\Windows\System32\kernel32.dll at 00007ffacbef0000
--- Process 14556 loaded C:\Windows\System32\KernelBase.dll at 00007ffaca420000
--- Process 14556 loaded C:\systems\cygwin64\usr\local\bin\cygruby260.dll at 00000005ed550000
--- Process 14556 loaded C:\systems\cygwin64\bin\cygwin1.dll at 0000000180040000
--- Process 14556 loaded C:\systems\cygwin64\bin\cyggcc_s-seh-1.dll at 00000003ff540000
--- Process 14556 loaded C:\Windows\System32\user32.dll at 00007ffacc910000
--- Process 14556 loaded C:\Windows\System32\gdi32.dll at 00007ffacc690000
--- Process 14556 loaded C:\Windows\System32\advapi32.dll at 00007ffacc1d0000
--- Process 14556 loaded C:\Windows\System32\msvcrt.dll at 00007ffaccc70000
--- Process 14556 loaded C:\Windows\System32\sechost.dll at 00007ffacc620000
--- Process 14556 loaded C:\Windows\System32\rpcrt4.dll at 00007ffaca6a0000
--- Process 14556 loaded C:\Windows\System32\sspicli.dll at 00007ffaca290000
--- Process 14556 loaded C:\Windows\System32\cryptbase.dll at 00007ffac9eb0000
--- Process 14556 loaded C:\Windows\System32\bcryptprimitives.dll at 00007ffac9e40000
--- Process 14556 loaded C:\Windows\System32\imm32.dll at 00007ffacc5e0000
--- Process 14556 loaded C:\Windows\System32\msctf.dll at 00007ffaca540000
--- Process 14556 loaded C:\systems\cygwin64\usr\local\lib\ruby\2.6.0\x86_64-cygwin\enc\encdb.so at 00000004b1d30000
--- Process 14556 loaded C:\systems\cygwin64\usr\local\lib\ruby\2.6.0\x86_64-cygwin\enc\trans\transdb.so at 000000040b2f0000
--- Process 14556 loaded C:\systems\cygwin64\usr\local\lib\ruby\2.6.0\x86_64-cygwin\stringio.so at 0000000587b40000
--- Process 14556 loaded C:\Windows\System32\authz.dll at 00007ffac9210000
--- Process 14556 thread 14032 created
--- Process 14556 thread 14032 exited with status 0x0
    1       1 [sig] ruby 21772 **********************************************
   84      85 [sig] ruby 21772 Program name: C:\systems\cygwin64\usr\local\bin\ruby.exe (pid 21772, ppid 632,windows pid 14556)
   50     135 [sig] ruby 21772 OS version:   Windows NT-6.3
   34     169 [sig] ruby 21772 **********************************************
This then also 'hangs', with CPU usage 0.0%. My understanding would be that this means that strace doesn't have much to show because there's nothing going on on the Ruby side.
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      - File strace_after_ctrl_C.txt strace_after_ctrl_C.txt added
duerst (Martin Dürst) wrote:
This then also 'hangs', with CPU usage 0.0%. My understanding would be that this means that
stracedoesn't have much to show because there's nothing going on on the Ruby side.
Attached is the rest of the strace output, after I hit Ctrl-C on the Ruby side. Not sure it will help anything.
        
           Updated by MSP-Greg (Greg L) about 7 years ago
          Updated by MSP-Greg (Greg L) about 7 years ago
          
          
        
        
      
      I thought that Cygwin & MSYS2/MinGW share some code (or may have some time ago). When running the code above, output as below:
ruby 2.6.0dev (2018-09-30 trunk 64889) [x64-mingw32]
Before sleeping
After sleeping, before method call
Before sleeping in method
After sleeping in method
After method call
FYI, Greg
        
           Updated by duerst (Martin Dürst) about 7 years ago
          Updated by duerst (Martin Dürst) about 7 years ago
          
          
        
        
      
      - Subject changed from open-uri hangs on cygwin to Nested Timeouts hang on cygwin
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      duerst@it.aoyama.ac.jp wrote:
File strace_after_ctrl_C.txt added
This then also 'hangs', with CPU usage 0.0%. My understanding would be that this means that
stracedoesn't have much to show because there's nothing going on on the Ruby side.Attached is the rest of the strace output, after I hit Ctrl-C on the Ruby side. Not sure it will help anything.
Thanks, but lots of stuff in there I don't understand since it's
not Linux...
Did Cygwin use sleepy timer-thread in Ruby 2.5 and earlier?
If not, can you try a 3rd patch to disable sleepy timer-thread?
(thread_win32.c doesn't use sleepy timer thread, either)
This 3rd patch goes on top of the previous two:
https://80x24.org/spew/20181015223141.18303-1-e@80x24.org/raw
The 3 patch series should be:
thread_pthread.c: fall back to UBF_TIMER_PTHREAD for Cygwin
thread_pthread.c: check and reset interrupts for UBF_TIMER_PTHREAD
thread_pthread.c: do not use sleepy timer thread on Cygwin
Footnote: Sleepy timer thread is ideal if your platform supports it
because it reduce CPU wakeups and therefore power consumption.
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      
    
        
           Updated by normalperson (Eric Wong) about 7 years ago
          Updated by normalperson (Eric Wong) about 7 years ago
          
          
        
        
      
      Martin: ping on [ruby-core:89410]
Thanks for the private reply, just wanted to make sure you were
aware of it.  I barely have time for Ruby in the past month
or so :x
        
           Updated by normalperson (Eric Wong) almost 7 years ago
          Updated by normalperson (Eric Wong) almost 7 years ago
          
          
        
        
      
      Ping again on https://bugs.ruby-lang.org/issues/15130
This 3rd patch goes on top of the previous two:
https://80x24.org/spew/20181015223141.18303-1-e@80x24.org/raw
The 3 patch series should be:
thread_pthread.c: fall back to UBF_TIMER_PTHREAD for Cygwin thread_pthread.c: check and reset interrupts for UBF_TIMER_PTHREAD thread_pthread.c: do not use sleepy timer thread on Cygwin
Btw, what is the reason for supporting Cygwin in addition to
other Windows builds?
It clutters thread_pthread.c a bit for *nix developers with
special cases.
        
           Updated by duerst (Martin Dürst) almost 7 years ago
          Updated by duerst (Martin Dürst) almost 7 years ago
          
          
        
        
      
      - Status changed from Open to Closed
First, I have good news. This error doesn't occur anymore. I discovered this yesterday when I was preparing to apply your third patch, and today I was able to confirm that it also applies without any of the patches you sent, on trunk. I very much hope this error doesn't resurface!
Many thanks for all your help and patience.
normalperson (Eric Wong) wrote:
Btw, what is the reason for supporting Cygwin in addition to
other Windows builds?
The simple answer is that same as for mountains: Because it's there.
A slightly longer answer is that for me (and my students), it's quite a good combination between the convenience of Windows and the power of Linux. Because of the current bug, I recently tried to install mingw/msys, but it was too confusing. Maybe it's just because I'm too used to cygwin, but maybe not. I try to avoid Microsoft stuff when I can avoid it, which applies to Visual Studio.
It clutters thread_pthread.c a bit for *nix developers with
special cases.
The world would be simpler but more boring if there were only one OS.
        
           Updated by duerst (Martin Dürst) almost 7 years ago
          Updated by duerst (Martin Dürst) almost 7 years ago
          
          
        
        
      
      Just for the record, the good news was at r65920. It may have started quite a bit before that, though.