cool.io 1.2.3-x86-mingw32 → 1.4.1-x86-mingw32
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.gitignore +3 -0
- data/.travis.yml +3 -3
- data/CHANGES.md +35 -0
- data/README.md +1 -3
- data/Rakefile +11 -13
- data/cool.io.gemspec +3 -2
- data/examples/callbacked_echo_server.rb +24 -0
- data/ext/cool.io/extconf.rb +8 -24
- data/ext/cool.io/loop.c +1 -1
- data/ext/iobuffer/iobuffer.c +2 -0
- data/ext/libev/Changes +123 -4
- data/ext/libev/LICENSE +2 -1
- data/ext/libev/README +8 -8
- data/ext/libev/ev.c +313 -144
- data/ext/libev/ev.h +18 -10
- data/ext/libev/ev_epoll.c +4 -1
- data/ext/libev/ev_kqueue.c +1 -1
- data/ext/libev/ev_select.c +3 -4
- data/ext/libev/ev_vars.h +3 -2
- data/ext/libev/ev_win32.c +1 -1
- data/ext/libev/win_select.patch +115 -0
- data/lib/cool.io.rb +6 -4
- data/lib/cool.io/dns_resolver.rb +4 -10
- data/lib/cool.io/dsl.rb +6 -2
- data/lib/cool.io/io.rb +36 -16
- data/lib/cool.io/loop.rb +3 -11
- data/lib/cool.io/meta.rb +2 -2
- data/lib/cool.io/version.rb +4 -2
- data/spec/async_watcher_spec.rb +5 -5
- data/spec/dns_spec.rb +11 -7
- data/spec/iobuffer_spec.rb +147 -0
- data/spec/spec_helper.rb +2 -2
- data/spec/stat_watcher_spec.rb +3 -3
- data/spec/tcp_server_spec.rb +98 -5
- data/spec/tcp_socket_spec.rb +185 -0
- data/spec/timer_watcher_spec.rb +23 -19
- data/spec/udp_socket_spec.rb +58 -0
- data/spec/unix_listener_spec.rb +7 -7
- data/spec/unix_server_spec.rb +7 -7
- metadata +83 -103
- data/examples/httpclient.rb +0 -38
- data/ext/http11_client/.gitignore +0 -5
- data/ext/http11_client/LICENSE +0 -31
- data/ext/http11_client/ext_help.h +0 -14
- data/ext/http11_client/extconf.rb +0 -6
- data/ext/http11_client/http11_client.c +0 -300
- data/ext/http11_client/http11_parser.c +0 -403
- data/ext/http11_client/http11_parser.h +0 -48
- data/ext/http11_client/http11_parser.rl +0 -173
- data/lib/cool.io/eventmachine.rb +0 -234
- data/lib/cool.io/http_client.rb +0 -427
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA1:
|
3
|
+
metadata.gz: b6642bffcf0b40c0f50f0e8429748e4deda9988b
|
4
|
+
data.tar.gz: 8d90a3f730addf0a28bfb5f4a4d5e2a39b8cc4fa
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 4599d50e5cae6df39e5ed72e0410eb322ffa36747d34c7fde13655dfca4831dfd2d95d9a5158fee846150c73cd478ec8241b52caf74e9371c4711e674189ac3e
|
7
|
+
data.tar.gz: d2c3277b5fce654234033ad3a97d60c94ab8e1b80d2ed8cc7300fed171ba89ac04ed933db919d1d94245530437af1deab9568b1cbc5c9bcf19e1cc5c280f6be6
|
data/.gitignore
CHANGED
data/.travis.yml
CHANGED
data/CHANGES.md
CHANGED
@@ -1,3 +1,38 @@
|
|
1
|
+
1.4.1
|
2
|
+
-----
|
3
|
+
|
4
|
+
* Use SleepEx instead of Sleep for better fix of process hung problem on windows environment
|
5
|
+
* Use rake-compiler-dock for cross compilation
|
6
|
+
|
7
|
+
1.4.0
|
8
|
+
-----
|
9
|
+
|
10
|
+
* Update libev to 4.20
|
11
|
+
* Sleep in timeout instead of select on Windows
|
12
|
+
|
13
|
+
1.3.1
|
14
|
+
-----
|
15
|
+
|
16
|
+
* Fix several bugs for JRuby support enhancement
|
17
|
+
* Fix deadlock bug on Windows environment
|
18
|
+
* Use RSpec3
|
19
|
+
|
20
|
+
1.3.0
|
21
|
+
-----
|
22
|
+
|
23
|
+
* Block evaluation doesn't change self for keeping consistency with Ruby block
|
24
|
+
* Remove EventMachine emulation module
|
25
|
+
* Remove HttpClient
|
26
|
+
* DSL syntax is no longer available by default. Need to require 'cool.io/dsl' in user code
|
27
|
+
* Update libev to 4.19
|
28
|
+
|
29
|
+
1.2.4
|
30
|
+
-----
|
31
|
+
|
32
|
+
* Fix a bug that #close for unconnected Socket doesn't detach all watchers (#33)
|
33
|
+
* Remove 1.8 support code
|
34
|
+
* Use standard library instead of own hosts list (#34)
|
35
|
+
|
1
36
|
1.2.3
|
2
37
|
-----
|
3
38
|
|
data/README.md
CHANGED
@@ -86,15 +86,13 @@ core socket classes are also provided. Among these are:
|
|
86
86
|
Cool.io::TCPSocket (or any subclass you wish to provide) whenever an incoming
|
87
87
|
connection is received.
|
88
88
|
|
89
|
-
* Cool.io::HttpClient - An HTTP/1.1 client with support for chunked encoding
|
90
|
-
and streaming response processing through asynchronous callbacks.
|
91
|
-
|
92
89
|
Example Program
|
93
90
|
---------------
|
94
91
|
|
95
92
|
Cool.io provides a Sinatra-like DSL for authoring event-driven programs:
|
96
93
|
|
97
94
|
require 'cool.io'
|
95
|
+
require 'cool.io/dsl'
|
98
96
|
|
99
97
|
ADDR = '127.0.0.1'
|
100
98
|
PORT = 4321
|
data/Rakefile
CHANGED
@@ -27,7 +27,7 @@ spec = eval(File.read("cool.io.gemspec"))
|
|
27
27
|
def configure_cross_compilation(ext)
|
28
28
|
unless RUBY_PLATFORM =~ /mswin|mingw/
|
29
29
|
ext.cross_compile = true
|
30
|
-
ext.cross_platform = '
|
30
|
+
ext.cross_platform = ['x86-mingw32', 'x64-mingw32']
|
31
31
|
end
|
32
32
|
end
|
33
33
|
|
@@ -36,23 +36,21 @@ Rake::ExtensionTask.new('iobuffer_ext', spec) do |ext|
|
|
36
36
|
configure_cross_compilation(ext)
|
37
37
|
end
|
38
38
|
|
39
|
-
Rake::ExtensionTask.new('http11_client', spec) do |ext|
|
40
|
-
ext.ext_dir = 'ext/http11_client'
|
41
|
-
configure_cross_compilation(ext)
|
42
|
-
end
|
43
|
-
|
44
39
|
Rake::ExtensionTask.new('cool.io_ext', spec) do |ext|
|
45
40
|
ext.ext_dir = 'ext/cool.io'
|
46
41
|
configure_cross_compilation(ext)
|
47
42
|
end
|
48
43
|
|
49
|
-
#
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
44
|
+
# Note that this rake-compiler-dock rake task dose not support bundle install(1) --path option.
|
45
|
+
# Please use bundle install instead when you execute this rake task.
|
46
|
+
namespace :build do
|
47
|
+
desc 'Build gems for Windows per rake-compiler-dock'
|
48
|
+
task :windows do
|
49
|
+
require 'rake_compiler_dock'
|
50
|
+
RakeCompilerDock.sh <<-CROSS
|
51
|
+
bundle
|
52
|
+
rake cross native gem RUBY_CC_VERSION='2.0.0:2.1.6:2.2.2'
|
53
|
+
CROSS
|
56
54
|
end
|
57
55
|
end
|
58
56
|
|
data/cool.io.gemspec
CHANGED
@@ -15,14 +15,15 @@ Gem::Specification.new do |s|
|
|
15
15
|
s.homepage = "http://coolio.github.com"
|
16
16
|
s.summary = "A cool framework for doing high performance I/O in Ruby"
|
17
17
|
s.description = "Cool.io provides a high performance event framework for Ruby which uses the libev C library"
|
18
|
-
s.extensions = ["ext/cool.io/extconf.rb", "ext/
|
18
|
+
s.extensions = ["ext/cool.io/extconf.rb", "ext/iobuffer/extconf.rb"]
|
19
19
|
|
20
20
|
s.files = `git ls-files`.split("\n")
|
21
21
|
s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
|
22
22
|
s.executables = `git ls-files -- bin/*`.split("\n").map{ |f| File.basename(f) }
|
23
23
|
s.require_paths = ["lib"]
|
24
24
|
|
25
|
-
s.add_development_dependency "rake-compiler", "~> 0.
|
25
|
+
s.add_development_dependency "rake-compiler", "~> 0.9.5"
|
26
|
+
s.add_development_dependency "rake-compiler-dock", "~> 0.4.3"
|
26
27
|
s.add_development_dependency "rspec", ">= 2.13.0"
|
27
28
|
s.add_development_dependency "rdoc", ">= 3.6.0"
|
28
29
|
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
$LOAD_PATH.unshift File.expand_path('../../lib', __FILE__)
|
2
|
+
|
3
|
+
require 'rubygems'
|
4
|
+
require 'cool.io'
|
5
|
+
|
6
|
+
ADDR = '127.0.0.1'
|
7
|
+
PORT = 4321
|
8
|
+
|
9
|
+
event_loop = Cool.io::Loop.default
|
10
|
+
server = Cool.io::TCPServer.new(ADDR, PORT) do |connection|
|
11
|
+
puts "#{connection.remote_addr}:#{connection.remote_port} connected"
|
12
|
+
|
13
|
+
connection.on_close do
|
14
|
+
puts "#{connection.remote_addr}:#{connection.remote_port} disconnected"
|
15
|
+
end
|
16
|
+
|
17
|
+
connection.on_read do |data|
|
18
|
+
connection.write data
|
19
|
+
end
|
20
|
+
end
|
21
|
+
server.attach(event_loop)
|
22
|
+
|
23
|
+
puts "Echo server listening on #{ADDR}:#{PORT}"
|
24
|
+
event_loop.run
|
data/ext/cool.io/extconf.rb
CHANGED
@@ -4,25 +4,11 @@ libs = []
|
|
4
4
|
|
5
5
|
$defs << "-DRUBY_VERSION_CODE=#{RUBY_VERSION.gsub(/\D/, '')}"
|
6
6
|
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
$defs << '-DHAVE_RB_THEREAD_CALL_WITHOUT_GVL'
|
13
|
-
end
|
14
|
-
|
15
|
-
if have_func('rb_thread_alone')
|
16
|
-
$defs << '-DHAVE_RB_THREAD_ALONE'
|
17
|
-
end
|
18
|
-
|
19
|
-
if have_func('rb_str_set_len')
|
20
|
-
$defs << '-DHAVE_RB_STR_SET_LEN'
|
21
|
-
end
|
22
|
-
|
23
|
-
if have_library('rt', 'clock_gettime')
|
24
|
-
libs << "-lrt"
|
25
|
-
end
|
7
|
+
have_func('rb_thread_blocking_region')
|
8
|
+
have_func('rb_thread_call_without_gvl')
|
9
|
+
have_func('rb_thread_alone')
|
10
|
+
have_func('rb_str_set_len')
|
11
|
+
have_library('rt', 'clock_gettime')
|
26
12
|
|
27
13
|
if have_header('sys/select.h')
|
28
14
|
$defs << '-DEV_USE_SELECT'
|
@@ -44,9 +30,7 @@ if have_header('port.h')
|
|
44
30
|
$defs << '-DEV_USE_PORT'
|
45
31
|
end
|
46
32
|
|
47
|
-
|
48
|
-
$defs << '-DHAVE_SYS_RESOURCE_H'
|
49
|
-
end
|
33
|
+
have_header('sys/resource.h')
|
50
34
|
|
51
35
|
# ncpu detection specifics
|
52
36
|
case RUBY_PLATFORM
|
@@ -64,14 +48,14 @@ dir_config('cool.io_ext')
|
|
64
48
|
create_makefile('cool.io_ext')
|
65
49
|
|
66
50
|
# win32 needs to link in "just the right order" for some reason or ioctlsocket will be mapped to an [inverted] ruby specific version. See libev mailing list for (not so helpful discussion--true cause I'm not sure, but this overcomes the symptom)
|
67
|
-
if RUBY_PLATFORM =~ /mingw|
|
51
|
+
if RUBY_PLATFORM =~ /mingw|mswin/
|
68
52
|
makefile_contents = File.read 'Makefile'
|
69
53
|
|
70
54
|
# "Init_cool could not be found" when loading cool.io.so.
|
71
55
|
# I'm not sure why this is needed. But this line causes "1114 A dynamic link library (DLL) initialization routine failed." So I commented out this line.
|
72
56
|
#makefile_contents.gsub! 'DLDFLAGS = ', 'DLDFLAGS = -export-all '
|
73
57
|
|
74
|
-
makefile_contents.gsub!
|
58
|
+
makefile_contents.gsub! /LIBS = (.*) (\S*ws2_32\S*)/i, 'LIBS = \\2 \\1'
|
75
59
|
File.open('Makefile', 'w') { |f| f.write makefile_contents }
|
76
60
|
end
|
77
61
|
|
data/ext/cool.io/loop.c
CHANGED
@@ -81,7 +81,7 @@ static void Coolio_Loop_free(struct Coolio_Loop *loop)
|
|
81
81
|
|
82
82
|
static VALUE Coolio_Loop_initialize(VALUE self)
|
83
83
|
{
|
84
|
-
Coolio_Loop_ev_loop_new(self, INT2NUM(0));
|
84
|
+
return Coolio_Loop_ev_loop_new(self, INT2NUM(0));
|
85
85
|
}
|
86
86
|
|
87
87
|
/* Wrapper for populating a Coolio_Loop struct with a new event loop */
|
data/ext/iobuffer/iobuffer.c
CHANGED
data/ext/libev/Changes
CHANGED
@@ -1,5 +1,125 @@
|
|
1
1
|
Revision history for libev, a high-performance and full-featured event loop.
|
2
2
|
|
3
|
+
TODO: ev_loop_wakeup
|
4
|
+
TODO: EV_STANDALONE == NO_HASSEL (do not use clock_gettime in ev_standalone)
|
5
|
+
TODO: faq, process a thing in each iteration
|
6
|
+
TODO: dbeugging tips, ev_verify, ev_init twice
|
7
|
+
TODO: ev_break for immediate exit (EVBREAK_NOW?)
|
8
|
+
TODO: ev_feed_child_event
|
9
|
+
TODO: document the special problem of signals around fork.
|
10
|
+
TODO: store pid for each signal
|
11
|
+
TODO: document file descriptor usage per loop
|
12
|
+
TODO: store loop pid_t and compare isndie signal handler,store 1 for same, 2 for differign pid, clean up in loop_fork
|
13
|
+
TODO: embed watchers need updating when fd changes
|
14
|
+
TODO: document portability requirements for atomic pointer access
|
15
|
+
TODO: possible cb aliasing?
|
16
|
+
TODO: document requirements for function pointers and calling conventions.
|
17
|
+
|
18
|
+
4.20 Sat Jun 20 13:01:43 CEST 2015
|
19
|
+
- prefer noexcept over throw () with C++ 11.
|
20
|
+
- update ecb.h due to incompatibilities with c11.
|
21
|
+
- fix a potential aliasing issue when reading and writing
|
22
|
+
watcher callbacks.
|
23
|
+
|
24
|
+
4.19 Thu Sep 25 08:18:25 CEST 2014
|
25
|
+
- ev.h wasn't valid C++ anymore, which tripped compilers other than
|
26
|
+
clang, msvc or gcc (analyzed by Raphael 'kena' Poss). Unfortunately,
|
27
|
+
C++ doesn't support typedefs for function pointers fully, so the affected
|
28
|
+
declarations have to spell out the types each time.
|
29
|
+
- when not using autoconf, tighten the check for clock_gettime and related
|
30
|
+
functionality.
|
31
|
+
|
32
|
+
4.18 Fri Sep 5 17:55:26 CEST 2014
|
33
|
+
- events on files were not always generated properly with the
|
34
|
+
epoll backend (testcase by Assaf Inbal).
|
35
|
+
- mark event pipe fd as cloexec after a fork (analyzed by Sami Farin).
|
36
|
+
- (ecb) support m68k, m88k and sh (patch by Miod Vallat).
|
37
|
+
- use a reasonable fallback for EV_NSIG instead of erroring out
|
38
|
+
when we can't detect the signal set size.
|
39
|
+
- in the absence of autoconf, do not use the clock syscall
|
40
|
+
on glibc >= 2.17 (avoids the syscall AND -lrt on systems
|
41
|
+
doing clock_gettime in userspace).
|
42
|
+
- ensure extern "C" function pointers are used for externally-visible
|
43
|
+
loop callbacks (not watcher callbacks yet).
|
44
|
+
- (ecb) work around memory barriers and volatile apparently both being
|
45
|
+
broken in visual studio 2008 and later (analysed and patch by Nicolas Noble).
|
46
|
+
|
47
|
+
4.15 Fri Mar 1 12:04:50 CET 2013
|
48
|
+
- destroying a non-default loop would stop the global waitpid
|
49
|
+
watcher (Denis Bilenko).
|
50
|
+
- queueing pending watchers of higher priority from a watcher now invokes
|
51
|
+
them in a timely fashion (reported by Denis Bilenko).
|
52
|
+
- add throw() to all libev functions that cannot throw exceptions, for
|
53
|
+
further code size decrease when compiling for C++.
|
54
|
+
- add throw () to callbacks that must not throw exceptions (allocator,
|
55
|
+
syserr, loop acquire/release, periodic reschedule cbs).
|
56
|
+
- fix event_base_loop return code, add event_get_callback, event_base_new,
|
57
|
+
event_base_get_method calls to improve libevent 1.x emulation and add
|
58
|
+
some libevent 2.x functionality (based on a patch by Jeff Davey).
|
59
|
+
- add more memory fences to fix a bug reported by Jeff Davey. Better
|
60
|
+
be overfenced than underprotected.
|
61
|
+
- ev_run now returns a boolean status (true meaning watchers are
|
62
|
+
still active).
|
63
|
+
- ev_once: undef EV_ERROR in ev_kqueue.c, to avoid clashing with
|
64
|
+
libev's EV_ERROR (reported by 191919).
|
65
|
+
- (ecb) add memory fence support for xlC (Darin McBride).
|
66
|
+
- (ecb) add memory fence support for gcc-mips (Anton Kirilov).
|
67
|
+
- (ecb) add memory fence support for gcc-alpha (Christian Weisgerber).
|
68
|
+
- work around some kernels losing file descriptors by leaking
|
69
|
+
the kqueue descriptor in the child.
|
70
|
+
- work around linux inotify not reporting IN_ATTRIB changes for directories
|
71
|
+
in many cases.
|
72
|
+
- include sys/syscall.h instead of plain syscall.h.
|
73
|
+
- check for io watcher loops in ev_verify, check for the most
|
74
|
+
common reported usage bug in ev_io_start.
|
75
|
+
- choose socket vs. WSASocket at compiletime using EV_USE_WSASOCKET.
|
76
|
+
- always use WSASend/WSARecv directly on windows, hoping that this
|
77
|
+
works in all cases (unlike read/write/send/recv...).
|
78
|
+
- try to detect signals around a fork faster (test program by
|
79
|
+
Denis Bilenko).
|
80
|
+
- work around recent glibc versions that leak memory in realloc.
|
81
|
+
- rename ev::embed::set to ev::embed::set_embed to avoid clashing
|
82
|
+
the watcher base set (loop) method.
|
83
|
+
- rewrite the async/signal pipe logic to always keep a valid fd, which
|
84
|
+
simplifies (and hopefully correctifies :) the race checking
|
85
|
+
on fork, at the cost of one extra fd.
|
86
|
+
- add fat, msdos, jffs2, ramfs, ntfs and btrfs to the list of
|
87
|
+
inotify-supporting filesystems.
|
88
|
+
- move orig_CFLAGS assignment to after AC_INIT, as newer autoconf
|
89
|
+
versions ignore it before
|
90
|
+
(https://bugzilla.redhat.com/show_bug.cgi?id=908096).
|
91
|
+
- add some untested android support.
|
92
|
+
- enum expressions must be of type int (reported by Juan Pablo L).
|
93
|
+
|
94
|
+
4.11 Sat Feb 4 19:52:39 CET 2012
|
95
|
+
- INCOMPATIBLE CHANGE: ev_timer_again now clears the pending status, as
|
96
|
+
was documented already, but not implemented in the repeating case.
|
97
|
+
- new compiletime symbols: EV_NO_SMP and EV_NO_THREADS.
|
98
|
+
- fix a race where the workaround against the epoll fork bugs
|
99
|
+
caused signals to not be handled anymore.
|
100
|
+
- correct backend_fudge for most backends, and implement a windows
|
101
|
+
specific workaround to avoid looping because we call both
|
102
|
+
select and Sleep, both with different time resolutions.
|
103
|
+
- document range and guarantees of ev_sleep.
|
104
|
+
- document reasonable ranges for periodics interval and offset.
|
105
|
+
- rename backend_fudge to backend_mintime to avoid future confusion :)
|
106
|
+
- change the default periodic reschedule function to hopefully be more
|
107
|
+
exact and correct even in corner cases or in the far future.
|
108
|
+
- do not rely on -lm anymore: use it when available but use our
|
109
|
+
own floor () if it is missing. This should make it easier to embed,
|
110
|
+
as no external libraries are required.
|
111
|
+
- strategically import macros from libecb and mark rarely-used functions
|
112
|
+
as cache-cold (saving almost 2k code size on typical amd64 setups).
|
113
|
+
- add Symbols.ev and Symbols.event files, that were missing.
|
114
|
+
- fix backend_mintime value for epoll (was 1/1024, is 1/1000 now).
|
115
|
+
- fix #3 "be smart about timeouts" to not "deadlock" when
|
116
|
+
timeout == now, also improve the section overall.
|
117
|
+
- avoid "AVOIDING FINISHING BEFORE RETURNING" idiom.
|
118
|
+
- support new EV_API_STATIC mode to make all libev symbols
|
119
|
+
static.
|
120
|
+
- supply default CFLAGS of -g -O3 with gcc when original CFLAGS
|
121
|
+
were empty.
|
122
|
+
|
3
123
|
4.04 Wed Feb 16 09:01:51 CET 2011
|
4
124
|
- fix two problems in the native win32 backend, where reuse of fd's
|
5
125
|
with different underlying handles caused handles not to be removed
|
@@ -94,7 +214,7 @@ Revision history for libev, a high-performance and full-featured event loop.
|
|
94
214
|
that this is a race condition regardless of EV_SIGNALFD.
|
95
215
|
- backport inotify code to C89.
|
96
216
|
- inotify file descriptors could leak into child processes.
|
97
|
-
- ev_stat watchers could keep an
|
217
|
+
- ev_stat watchers could keep an erroneous extra ref on the loop,
|
98
218
|
preventing exit when unregistering all watchers (testcases
|
99
219
|
provided by ry@tinyclouds.org).
|
100
220
|
- implement EV_WIN32_HANDLE_TO_FD and EV_WIN32_CLOSE_FD configuration
|
@@ -162,8 +282,8 @@ Revision history for libev, a high-performance and full-featured event loop.
|
|
162
282
|
Malek Hadj-Ali).
|
163
283
|
- implement ev_suspend and ev_resume.
|
164
284
|
- new EV_CUSTOM revents flag for use by applications.
|
165
|
-
- add documentation section about
|
166
|
-
- add a glossary to the
|
285
|
+
- add documentation section about priorities.
|
286
|
+
- add a glossary to the documentation.
|
167
287
|
- extend the ev_fork description slightly.
|
168
288
|
- optimize a jump out of call_pending.
|
169
289
|
|
@@ -385,4 +505,3 @@ Revision history for libev, a high-performance and full-featured event loop.
|
|
385
505
|
|
386
506
|
0.1 Wed Oct 31 21:31:48 CET 2007
|
387
507
|
- original version; hacked together in <24h.
|
388
|
-
|
data/ext/libev/LICENSE
CHANGED
@@ -1,4 +1,5 @@
|
|
1
|
-
All files in libev are
|
1
|
+
All files in libev are
|
2
|
+
Copyright (c)2007,2008,2009,2010,2011,2012,2013 Marc Alexander Lehmann.
|
2
3
|
|
3
4
|
Redistribution and use in source and binary forms, with or without
|
4
5
|
modification, are permitted provided that the following conditions are
|
data/ext/libev/README
CHANGED
@@ -24,23 +24,23 @@ ABOUT
|
|
24
24
|
- relative timers/timeouts (handle time jumps).
|
25
25
|
- fast intra-thread communication between multiple
|
26
26
|
event loops (with optional fast linux eventfd backend).
|
27
|
-
- extremely easy to embed
|
28
|
-
|
27
|
+
- extremely easy to embed (fully documented, no dependencies,
|
28
|
+
autoconf supported but optional).
|
29
|
+
- very small codebase, no bloated library, simple code.
|
29
30
|
- fully extensible by being able to plug into the event loop,
|
30
31
|
integrate other event loops, integrate other event loop users.
|
31
32
|
- very little memory use (small watchers, small event loop data).
|
32
33
|
- optional C++ interface allowing method and function callbacks
|
33
34
|
at no extra memory or runtime overhead.
|
34
35
|
- optional Perl interface with similar characteristics (capable
|
35
|
-
of running Glib/Gtk2 on libev
|
36
|
-
libadns).
|
36
|
+
of running Glib/Gtk2 on libev).
|
37
37
|
- support for other languages (multiple C++ interfaces, D, Ruby,
|
38
38
|
Python) available from third-parties.
|
39
39
|
|
40
|
-
Examples of programs that embed libev: the EV perl module,
|
41
|
-
rxvt-unicode, gvpe (GNU Virtual Private Ethernet), the
|
42
|
-
server (http://www.deliantra.net/), Rubinius (a
|
43
|
-
VM), the Ebb web server, the Rev event toolkit.
|
40
|
+
Examples of programs that embed libev: the EV perl module, node.js,
|
41
|
+
auditd, rxvt-unicode, gvpe (GNU Virtual Private Ethernet), the
|
42
|
+
Deliantra MMORPG server (http://www.deliantra.net/), Rubinius (a
|
43
|
+
next-generation Ruby VM), the Ebb web server, the Rev event toolkit.
|
44
44
|
|
45
45
|
|
46
46
|
CONTRIBUTORS
|
data/ext/libev/ev.c
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
/*
|
2
2
|
* libev event processing core, watcher management
|
3
3
|
*
|
4
|
-
* Copyright (c) 2007,2008,2009,2010,2011,2012 Marc Alexander Lehmann <libev@schmorp.de>
|
4
|
+
* Copyright (c) 2007,2008,2009,2010,2011,2012,2013 Marc Alexander Lehmann <libev@schmorp.de>
|
5
5
|
* All rights reserved.
|
6
6
|
*
|
7
7
|
* Redistribution and use in source and binary forms, with or without modifica-
|
@@ -49,11 +49,11 @@
|
|
49
49
|
# include "config.h"
|
50
50
|
# endif
|
51
51
|
|
52
|
-
#if HAVE_FLOOR
|
53
|
-
#
|
54
|
-
#
|
52
|
+
# if HAVE_FLOOR
|
53
|
+
# ifndef EV_USE_FLOOR
|
54
|
+
# define EV_USE_FLOOR 1
|
55
|
+
# endif
|
55
56
|
# endif
|
56
|
-
#endif
|
57
57
|
|
58
58
|
# if HAVE_CLOCK_SYSCALL
|
59
59
|
# ifndef EV_USE_CLOCK_SYSCALL
|
@@ -111,7 +111,7 @@
|
|
111
111
|
# undef EV_USE_POLL
|
112
112
|
# define EV_USE_POLL 0
|
113
113
|
# endif
|
114
|
-
|
114
|
+
|
115
115
|
# if HAVE_EPOLL_CTL && HAVE_SYS_EPOLL_H
|
116
116
|
# ifndef EV_USE_EPOLL
|
117
117
|
# define EV_USE_EPOLL EV_FEATURE_BACKENDS
|
@@ -120,7 +120,7 @@
|
|
120
120
|
# undef EV_USE_EPOLL
|
121
121
|
# define EV_USE_EPOLL 0
|
122
122
|
# endif
|
123
|
-
|
123
|
+
|
124
124
|
# if HAVE_KQUEUE && HAVE_SYS_EVENT_H
|
125
125
|
# ifndef EV_USE_KQUEUE
|
126
126
|
# define EV_USE_KQUEUE EV_FEATURE_BACKENDS
|
@@ -129,7 +129,7 @@
|
|
129
129
|
# undef EV_USE_KQUEUE
|
130
130
|
# define EV_USE_KQUEUE 0
|
131
131
|
# endif
|
132
|
-
|
132
|
+
|
133
133
|
# if HAVE_PORT_H && HAVE_PORT_CREATE
|
134
134
|
# ifndef EV_USE_PORT
|
135
135
|
# define EV_USE_PORT EV_FEATURE_BACKENDS
|
@@ -165,7 +165,7 @@
|
|
165
165
|
# undef EV_USE_EVENTFD
|
166
166
|
# define EV_USE_EVENTFD 0
|
167
167
|
# endif
|
168
|
-
|
168
|
+
|
169
169
|
#endif
|
170
170
|
|
171
171
|
#include <stdlib.h>
|
@@ -247,10 +247,7 @@
|
|
247
247
|
#elif defined _sys_nsig
|
248
248
|
# define EV_NSIG (_sys_nsig) /* Solaris 2.5 */
|
249
249
|
#else
|
250
|
-
#
|
251
|
-
/* to make it compile regardless, just remove the above line, */
|
252
|
-
/* but consider reporting it, too! :) */
|
253
|
-
# define EV_NSIG 65
|
250
|
+
# define EV_NSIG (8 * sizeof (sigset_t) + 1)
|
254
251
|
#endif
|
255
252
|
|
256
253
|
#ifndef EV_USE_FLOOR
|
@@ -258,13 +255,22 @@
|
|
258
255
|
#endif
|
259
256
|
|
260
257
|
#ifndef EV_USE_CLOCK_SYSCALL
|
261
|
-
# if __linux && __GLIBC__
|
258
|
+
# if __linux && __GLIBC__ == 2 && __GLIBC_MINOR__ < 17
|
262
259
|
# define EV_USE_CLOCK_SYSCALL EV_FEATURE_OS
|
263
260
|
# else
|
264
261
|
# define EV_USE_CLOCK_SYSCALL 0
|
265
262
|
# endif
|
266
263
|
#endif
|
267
264
|
|
265
|
+
#if !(_POSIX_TIMERS > 0)
|
266
|
+
# ifndef EV_USE_MONOTONIC
|
267
|
+
# define EV_USE_MONOTONIC 0
|
268
|
+
# endif
|
269
|
+
# ifndef EV_USE_REALTIME
|
270
|
+
# define EV_USE_REALTIME 0
|
271
|
+
# endif
|
272
|
+
#endif
|
273
|
+
|
268
274
|
#ifndef EV_USE_MONOTONIC
|
269
275
|
# if defined _POSIX_MONOTONIC_CLOCK && _POSIX_MONOTONIC_CLOCK >= 0
|
270
276
|
# define EV_USE_MONOTONIC EV_FEATURE_OS
|
@@ -491,7 +497,7 @@ struct signalfd_siginfo
|
|
491
497
|
/*
|
492
498
|
* libecb - http://software.schmorp.de/pkg/libecb
|
493
499
|
*
|
494
|
-
* Copyright (©) 2009-
|
500
|
+
* Copyright (©) 2009-2015 Marc Alexander Lehmann <libecb@schmorp.de>
|
495
501
|
* Copyright (©) 2011 Emanuele Giaquinta
|
496
502
|
* All rights reserved.
|
497
503
|
*
|
@@ -515,13 +521,24 @@ struct signalfd_siginfo
|
|
515
521
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTH-
|
516
522
|
* ERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
517
523
|
* OF THE POSSIBILITY OF SUCH DAMAGE.
|
524
|
+
*
|
525
|
+
* Alternatively, the contents of this file may be used under the terms of
|
526
|
+
* the GNU General Public License ("GPL") version 2 or any later version,
|
527
|
+
* in which case the provisions of the GPL are applicable instead of
|
528
|
+
* the above. If you wish to allow the use of your version of this file
|
529
|
+
* only under the terms of the GPL and not to allow others to use your
|
530
|
+
* version of this file under the BSD license, indicate your decision
|
531
|
+
* by deleting the provisions above and replace them with the notice
|
532
|
+
* and other provisions required by the GPL. If you do not delete the
|
533
|
+
* provisions above, a recipient may use your version of this file under
|
534
|
+
* either the BSD or the GPL.
|
518
535
|
*/
|
519
536
|
|
520
537
|
#ifndef ECB_H
|
521
538
|
#define ECB_H
|
522
539
|
|
523
540
|
/* 16 bits major, 16 bits minor */
|
524
|
-
#define ECB_VERSION
|
541
|
+
#define ECB_VERSION 0x00010004
|
525
542
|
|
526
543
|
#ifdef _WIN32
|
527
544
|
typedef signed char int8_t;
|
@@ -555,9 +572,12 @@ struct signalfd_siginfo
|
|
555
572
|
#endif
|
556
573
|
#endif
|
557
574
|
|
575
|
+
#define ECB_GCC_AMD64 (__amd64 || __amd64__ || __x86_64 || __x86_64__)
|
576
|
+
#define ECB_MSVC_AMD64 (_M_AMD64 || _M_X64)
|
577
|
+
|
558
578
|
/* work around x32 idiocy by defining proper macros */
|
559
|
-
#if
|
560
|
-
#if
|
579
|
+
#if ECB_GCC_AMD64 || ECB_MSVC_AMD64
|
580
|
+
#if _ILP32
|
561
581
|
#define ECB_AMD64_X32 1
|
562
582
|
#else
|
563
583
|
#define ECB_AMD64 1
|
@@ -571,20 +591,40 @@ struct signalfd_siginfo
|
|
571
591
|
* we try to detect these and simply assume they are not gcc - if they have
|
572
592
|
* an issue with that they should have done it right in the first place.
|
573
593
|
*/
|
574
|
-
#
|
575
|
-
#
|
576
|
-
|
577
|
-
#
|
578
|
-
|
579
|
-
|
594
|
+
#if !defined __GNUC_MINOR__ || defined __INTEL_COMPILER || defined __SUNPRO_C || defined __SUNPRO_CC || defined __llvm__ || defined __clang__
|
595
|
+
#define ECB_GCC_VERSION(major,minor) 0
|
596
|
+
#else
|
597
|
+
#define ECB_GCC_VERSION(major,minor) (__GNUC__ > (major) || (__GNUC__ == (major) && __GNUC_MINOR__ >= (minor)))
|
598
|
+
#endif
|
599
|
+
|
600
|
+
#define ECB_CLANG_VERSION(major,minor) (__clang_major__ > (major) || (__clang_major__ == (major) && __clang_minor__ >= (minor)))
|
601
|
+
|
602
|
+
#if __clang__ && defined __has_builtin
|
603
|
+
#define ECB_CLANG_BUILTIN(x) __has_builtin (x)
|
604
|
+
#else
|
605
|
+
#define ECB_CLANG_BUILTIN(x) 0
|
606
|
+
#endif
|
607
|
+
|
608
|
+
#if __clang__ && defined __has_extension
|
609
|
+
#define ECB_CLANG_EXTENSION(x) __has_extension (x)
|
610
|
+
#else
|
611
|
+
#define ECB_CLANG_EXTENSION(x) 0
|
580
612
|
#endif
|
581
613
|
|
582
|
-
#define ECB_C (__STDC__+0) /* this assumes that __STDC__ is either empty or a number */
|
583
|
-
#define ECB_C99 (__STDC_VERSION__ >= 199901L)
|
584
|
-
#define ECB_C11 (__STDC_VERSION__ >= 201112L)
|
585
614
|
#define ECB_CPP (__cplusplus+0)
|
586
615
|
#define ECB_CPP11 (__cplusplus >= 201103L)
|
587
616
|
|
617
|
+
#if ECB_CPP
|
618
|
+
#define ECB_C 0
|
619
|
+
#define ECB_STDC_VERSION 0
|
620
|
+
#else
|
621
|
+
#define ECB_C 1
|
622
|
+
#define ECB_STDC_VERSION __STDC_VERSION__
|
623
|
+
#endif
|
624
|
+
|
625
|
+
#define ECB_C99 (ECB_STDC_VERSION >= 199901L)
|
626
|
+
#define ECB_C11 (ECB_STDC_VERSION >= 201112L)
|
627
|
+
|
588
628
|
#if ECB_CPP
|
589
629
|
#define ECB_EXTERN_C extern "C"
|
590
630
|
#define ECB_EXTERN_C_BEG ECB_EXTERN_C {
|
@@ -608,13 +648,18 @@ struct signalfd_siginfo
|
|
608
648
|
#define ECB_MEMORY_FENCE do { } while (0)
|
609
649
|
#endif
|
610
650
|
|
651
|
+
/* http://www-01.ibm.com/support/knowledgecenter/SSGH3R_13.1.0/com.ibm.xlcpp131.aix.doc/compiler_ref/compiler_builtins.html */
|
652
|
+
#if __xlC__ && ECB_CPP
|
653
|
+
#include <builtins.h>
|
654
|
+
#endif
|
655
|
+
|
611
656
|
#ifndef ECB_MEMORY_FENCE
|
612
657
|
#if ECB_GCC_VERSION(2,5) || defined __INTEL_COMPILER || (__llvm__ && __GNUC__) || __SUNPRO_C >= 0x5110 || __SUNPRO_CC >= 0x5110
|
613
658
|
#if __i386 || __i386__
|
614
659
|
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("lock; orb $0, -1(%%esp)" : : : "memory")
|
615
660
|
#define ECB_MEMORY_FENCE_ACQUIRE __asm__ __volatile__ ("" : : : "memory")
|
616
661
|
#define ECB_MEMORY_FENCE_RELEASE __asm__ __volatile__ ("")
|
617
|
-
#elif
|
662
|
+
#elif ECB_GCC_AMD64
|
618
663
|
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("mfence" : : : "memory")
|
619
664
|
#define ECB_MEMORY_FENCE_ACQUIRE __asm__ __volatile__ ("" : : : "memory")
|
620
665
|
#define ECB_MEMORY_FENCE_RELEASE __asm__ __volatile__ ("")
|
@@ -626,14 +671,18 @@ struct signalfd_siginfo
|
|
626
671
|
#elif defined __ARM_ARCH_7__ || defined __ARM_ARCH_7A__ \
|
627
672
|
|| defined __ARM_ARCH_7M__ || defined __ARM_ARCH_7R__
|
628
673
|
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("dmb" : : : "memory")
|
629
|
-
#elif
|
674
|
+
#elif __aarch64__
|
675
|
+
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("dmb ish" : : : "memory")
|
676
|
+
#elif (__sparc || __sparc__) && !__sparcv8
|
630
677
|
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("membar #LoadStore | #LoadLoad | #StoreStore | #StoreLoad" : : : "memory")
|
631
678
|
#define ECB_MEMORY_FENCE_ACQUIRE __asm__ __volatile__ ("membar #LoadStore | #LoadLoad" : : : "memory")
|
632
679
|
#define ECB_MEMORY_FENCE_RELEASE __asm__ __volatile__ ("membar #LoadStore | #StoreStore")
|
633
680
|
#elif defined __s390__ || defined __s390x__
|
634
681
|
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("bcr 15,0" : : : "memory")
|
635
682
|
#elif defined __mips__
|
636
|
-
|
683
|
+
/* GNU/Linux emulates sync on mips1 architectures, so we force its use */
|
684
|
+
/* anybody else who still uses mips1 is supposed to send in their version, with detection code. */
|
685
|
+
#define ECB_MEMORY_FENCE __asm__ __volatile__ (".set mips2; sync; .set mips0" : : : "memory")
|
637
686
|
#elif defined __alpha__
|
638
687
|
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("mb" : : : "memory")
|
639
688
|
#elif defined __hppa__
|
@@ -641,6 +690,12 @@ struct signalfd_siginfo
|
|
641
690
|
#define ECB_MEMORY_FENCE_RELEASE __asm__ __volatile__ ("")
|
642
691
|
#elif defined __ia64__
|
643
692
|
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("mf" : : : "memory")
|
693
|
+
#elif defined __m68k__
|
694
|
+
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("" : : : "memory")
|
695
|
+
#elif defined __m88k__
|
696
|
+
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("tb1 0,%%r0,128" : : : "memory")
|
697
|
+
#elif defined __sh__
|
698
|
+
#define ECB_MEMORY_FENCE __asm__ __volatile__ ("" : : : "memory")
|
644
699
|
#endif
|
645
700
|
#endif
|
646
701
|
#endif
|
@@ -649,18 +704,23 @@ struct signalfd_siginfo
|
|
649
704
|
#if ECB_GCC_VERSION(4,7)
|
650
705
|
/* see comment below (stdatomic.h) about the C11 memory model. */
|
651
706
|
#define ECB_MEMORY_FENCE __atomic_thread_fence (__ATOMIC_SEQ_CST)
|
707
|
+
#define ECB_MEMORY_FENCE_ACQUIRE __atomic_thread_fence (__ATOMIC_ACQUIRE)
|
708
|
+
#define ECB_MEMORY_FENCE_RELEASE __atomic_thread_fence (__ATOMIC_RELEASE)
|
652
709
|
|
653
|
-
|
654
|
-
|
655
|
-
|
656
|
-
|
657
|
-
|
658
|
-
* // see comment below (stdatomic.h) about the C11 memory model.
|
659
|
-
* #define ECB_MEMORY_FENCE __c11_atomic_thread_fence (__ATOMIC_SEQ_CST)
|
660
|
-
*/
|
710
|
+
#elif ECB_CLANG_EXTENSION(c_atomic)
|
711
|
+
/* see comment below (stdatomic.h) about the C11 memory model. */
|
712
|
+
#define ECB_MEMORY_FENCE __c11_atomic_thread_fence (__ATOMIC_SEQ_CST)
|
713
|
+
#define ECB_MEMORY_FENCE_ACQUIRE __c11_atomic_thread_fence (__ATOMIC_ACQUIRE)
|
714
|
+
#define ECB_MEMORY_FENCE_RELEASE __c11_atomic_thread_fence (__ATOMIC_RELEASE)
|
661
715
|
|
662
716
|
#elif ECB_GCC_VERSION(4,4) || defined __INTEL_COMPILER || defined __clang__
|
663
717
|
#define ECB_MEMORY_FENCE __sync_synchronize ()
|
718
|
+
#elif _MSC_VER >= 1500 /* VC++ 2008 */
|
719
|
+
/* apparently, microsoft broke all the memory barrier stuff in Visual Studio 2008... */
|
720
|
+
#pragma intrinsic(_ReadBarrier,_WriteBarrier,_ReadWriteBarrier)
|
721
|
+
#define ECB_MEMORY_FENCE _ReadWriteBarrier (); MemoryBarrier()
|
722
|
+
#define ECB_MEMORY_FENCE_ACQUIRE _ReadWriteBarrier (); MemoryBarrier() /* according to msdn, _ReadBarrier is not a load fence */
|
723
|
+
#define ECB_MEMORY_FENCE_RELEASE _WriteBarrier (); MemoryBarrier()
|
664
724
|
#elif _MSC_VER >= 1400 /* VC++ 2005 */
|
665
725
|
#pragma intrinsic(_ReadBarrier,_WriteBarrier,_ReadWriteBarrier)
|
666
726
|
#define ECB_MEMORY_FENCE _ReadWriteBarrier ()
|
@@ -690,6 +750,8 @@ struct signalfd_siginfo
|
|
690
750
|
/* for most usages, or gcc and clang have a bug */
|
691
751
|
/* I *currently* lean towards the latter, and inefficiently implement */
|
692
752
|
/* all three of ecb's fences as a seq_cst fence */
|
753
|
+
/* Update, gcc-4.8 generates mfence for all c++ fences, but nothing */
|
754
|
+
/* for all __atomic_thread_fence's except seq_cst */
|
693
755
|
#define ECB_MEMORY_FENCE atomic_thread_fence (memory_order_seq_cst)
|
694
756
|
#endif
|
695
757
|
#endif
|
@@ -722,7 +784,7 @@ struct signalfd_siginfo
|
|
722
784
|
|
723
785
|
/*****************************************************************************/
|
724
786
|
|
725
|
-
#if
|
787
|
+
#if ECB_CPP
|
726
788
|
#define ecb_inline static inline
|
727
789
|
#elif ECB_GCC_VERSION(2,5)
|
728
790
|
#define ecb_inline static __inline__
|
@@ -746,35 +808,79 @@ typedef int ecb_bool;
|
|
746
808
|
#define ECB_CONCAT(a, b) ECB_CONCAT_(a, b)
|
747
809
|
#define ECB_STRINGIFY_(a) # a
|
748
810
|
#define ECB_STRINGIFY(a) ECB_STRINGIFY_(a)
|
811
|
+
#define ECB_STRINGIFY_EXPR(expr) ((expr), ECB_STRINGIFY_ (expr))
|
749
812
|
|
750
813
|
#define ecb_function_ ecb_inline
|
751
814
|
|
752
|
-
#if ECB_GCC_VERSION(3,1)
|
753
|
-
#define ecb_attribute(attrlist) __attribute__(attrlist)
|
754
|
-
#define ecb_is_constant(expr) __builtin_constant_p (expr)
|
755
|
-
#define ecb_expect(expr,value) __builtin_expect ((expr),(value))
|
756
|
-
#define ecb_prefetch(addr,rw,locality) __builtin_prefetch (addr, rw, locality)
|
815
|
+
#if ECB_GCC_VERSION(3,1) || ECB_CLANG_VERSION(2,8)
|
816
|
+
#define ecb_attribute(attrlist) __attribute__ (attrlist)
|
757
817
|
#else
|
758
818
|
#define ecb_attribute(attrlist)
|
819
|
+
#endif
|
820
|
+
|
821
|
+
#if ECB_GCC_VERSION(3,1) || ECB_CLANG_BUILTIN(__builtin_constant_p)
|
822
|
+
#define ecb_is_constant(expr) __builtin_constant_p (expr)
|
823
|
+
#else
|
824
|
+
/* possible C11 impl for integral types
|
825
|
+
typedef struct ecb_is_constant_struct ecb_is_constant_struct;
|
826
|
+
#define ecb_is_constant(expr) _Generic ((1 ? (struct ecb_is_constant_struct *)0 : (void *)((expr) - (expr)), ecb_is_constant_struct *: 0, default: 1)) */
|
827
|
+
|
759
828
|
#define ecb_is_constant(expr) 0
|
829
|
+
#endif
|
830
|
+
|
831
|
+
#if ECB_GCC_VERSION(3,1) || ECB_CLANG_BUILTIN(__builtin_expect)
|
832
|
+
#define ecb_expect(expr,value) __builtin_expect ((expr),(value))
|
833
|
+
#else
|
760
834
|
#define ecb_expect(expr,value) (expr)
|
835
|
+
#endif
|
836
|
+
|
837
|
+
#if ECB_GCC_VERSION(3,1) || ECB_CLANG_BUILTIN(__builtin_prefetch)
|
838
|
+
#define ecb_prefetch(addr,rw,locality) __builtin_prefetch (addr, rw, locality)
|
839
|
+
#else
|
761
840
|
#define ecb_prefetch(addr,rw,locality)
|
762
841
|
#endif
|
763
842
|
|
764
843
|
/* no emulation for ecb_decltype */
|
765
|
-
#if
|
766
|
-
|
767
|
-
|
768
|
-
#define ecb_decltype(x)
|
844
|
+
#if ECB_CPP11
|
845
|
+
// older implementations might have problems with decltype(x)::type, work around it
|
846
|
+
template<class T> struct ecb_decltype_t { typedef T type; };
|
847
|
+
#define ecb_decltype(x) ecb_decltype_t<decltype (x)>::type
|
848
|
+
#elif ECB_GCC_VERSION(3,0) || ECB_CLANG_VERSION(2,8)
|
849
|
+
#define ecb_decltype(x) __typeof__ (x)
|
850
|
+
#endif
|
851
|
+
|
852
|
+
#if _MSC_VER >= 1300
|
853
|
+
#define ecb_deprecated __declspec (deprecated)
|
854
|
+
#else
|
855
|
+
#define ecb_deprecated ecb_attribute ((__deprecated__))
|
856
|
+
#endif
|
857
|
+
|
858
|
+
#if _MSC_VER >= 1500
|
859
|
+
#define ecb_deprecated_message(msg) __declspec (deprecated (msg))
|
860
|
+
#elif ECB_GCC_VERSION(4,5)
|
861
|
+
#define ecb_deprecated_message(msg) ecb_attribute ((__deprecated__ (msg))
|
862
|
+
#else
|
863
|
+
#define ecb_deprecated_message(msg) ecb_deprecated
|
864
|
+
#endif
|
865
|
+
|
866
|
+
#if _MSC_VER >= 1400
|
867
|
+
#define ecb_noinline __declspec (noinline)
|
868
|
+
#else
|
869
|
+
#define ecb_noinline ecb_attribute ((__noinline__))
|
769
870
|
#endif
|
770
871
|
|
771
|
-
#define ecb_noinline ecb_attribute ((__noinline__))
|
772
872
|
#define ecb_unused ecb_attribute ((__unused__))
|
773
873
|
#define ecb_const ecb_attribute ((__const__))
|
774
874
|
#define ecb_pure ecb_attribute ((__pure__))
|
775
875
|
|
776
|
-
#if ECB_C11
|
876
|
+
#if ECB_C11 || __IBMC_NORETURN
|
877
|
+
/* http://www-01.ibm.com/support/knowledgecenter/SSGH3R_13.1.0/com.ibm.xlcpp131.aix.doc/language_ref/noreturn.html */
|
777
878
|
#define ecb_noreturn _Noreturn
|
879
|
+
#elif ECB_CPP11
|
880
|
+
#define ecb_noreturn [[noreturn]]
|
881
|
+
#elif _MSC_VER >= 1200
|
882
|
+
/* http://msdn.microsoft.com/en-us/library/k6ktzx3s.aspx */
|
883
|
+
#define ecb_noreturn __declspec (noreturn)
|
778
884
|
#else
|
779
885
|
#define ecb_noreturn ecb_attribute ((__noreturn__))
|
780
886
|
#endif
|
@@ -799,7 +905,10 @@ typedef int ecb_bool;
|
|
799
905
|
#define ecb_unlikely(expr) ecb_expect_false (expr)
|
800
906
|
|
801
907
|
/* count trailing zero bits and count # of one bits */
|
802
|
-
#if ECB_GCC_VERSION(3,4)
|
908
|
+
#if ECB_GCC_VERSION(3,4) \
|
909
|
+
|| (ECB_CLANG_BUILTIN(__builtin_clz) && ECB_CLANG_BUILTIN(__builtin_clzll) \
|
910
|
+
&& ECB_CLANG_BUILTIN(__builtin_ctz) && ECB_CLANG_BUILTIN(__builtin_ctzll) \
|
911
|
+
&& ECB_CLANG_BUILTIN(__builtin_popcount))
|
803
912
|
/* we assume int == 32 bit, long == 32 or 64 bit and long long == 64 bit */
|
804
913
|
#define ecb_ld32(x) (__builtin_clz (x) ^ 31)
|
805
914
|
#define ecb_ld64(x) (__builtin_clzll (x) ^ 63)
|
@@ -808,8 +917,8 @@ typedef int ecb_bool;
|
|
808
917
|
#define ecb_popcount32(x) __builtin_popcount (x)
|
809
918
|
/* no popcountll */
|
810
919
|
#else
|
811
|
-
ecb_function_ int ecb_ctz32 (uint32_t x)
|
812
|
-
ecb_function_ int
|
920
|
+
ecb_function_ ecb_const int ecb_ctz32 (uint32_t x);
|
921
|
+
ecb_function_ ecb_const int
|
813
922
|
ecb_ctz32 (uint32_t x)
|
814
923
|
{
|
815
924
|
int r = 0;
|
@@ -833,16 +942,16 @@ typedef int ecb_bool;
|
|
833
942
|
return r;
|
834
943
|
}
|
835
944
|
|
836
|
-
ecb_function_ int ecb_ctz64 (uint64_t x)
|
837
|
-
ecb_function_ int
|
945
|
+
ecb_function_ ecb_const int ecb_ctz64 (uint64_t x);
|
946
|
+
ecb_function_ ecb_const int
|
838
947
|
ecb_ctz64 (uint64_t x)
|
839
948
|
{
|
840
949
|
int shift = x & 0xffffffffU ? 0 : 32;
|
841
950
|
return ecb_ctz32 (x >> shift) + shift;
|
842
951
|
}
|
843
952
|
|
844
|
-
ecb_function_ int ecb_popcount32 (uint32_t x)
|
845
|
-
ecb_function_ int
|
953
|
+
ecb_function_ ecb_const int ecb_popcount32 (uint32_t x);
|
954
|
+
ecb_function_ ecb_const int
|
846
955
|
ecb_popcount32 (uint32_t x)
|
847
956
|
{
|
848
957
|
x -= (x >> 1) & 0x55555555;
|
@@ -853,8 +962,8 @@ typedef int ecb_bool;
|
|
853
962
|
return x >> 24;
|
854
963
|
}
|
855
964
|
|
856
|
-
ecb_function_ int ecb_ld32 (uint32_t x)
|
857
|
-
ecb_function_ int ecb_ld32 (uint32_t x)
|
965
|
+
ecb_function_ ecb_const int ecb_ld32 (uint32_t x);
|
966
|
+
ecb_function_ ecb_const int ecb_ld32 (uint32_t x)
|
858
967
|
{
|
859
968
|
int r = 0;
|
860
969
|
|
@@ -867,8 +976,8 @@ typedef int ecb_bool;
|
|
867
976
|
return r;
|
868
977
|
}
|
869
978
|
|
870
|
-
ecb_function_ int ecb_ld64 (uint64_t x)
|
871
|
-
ecb_function_ int ecb_ld64 (uint64_t x)
|
979
|
+
ecb_function_ ecb_const int ecb_ld64 (uint64_t x);
|
980
|
+
ecb_function_ ecb_const int ecb_ld64 (uint64_t x)
|
872
981
|
{
|
873
982
|
int r = 0;
|
874
983
|
|
@@ -878,20 +987,20 @@ typedef int ecb_bool;
|
|
878
987
|
}
|
879
988
|
#endif
|
880
989
|
|
881
|
-
ecb_function_ ecb_bool ecb_is_pot32 (uint32_t x)
|
882
|
-
ecb_function_ ecb_bool ecb_is_pot32 (uint32_t x) { return !(x & (x - 1)); }
|
883
|
-
ecb_function_ ecb_bool ecb_is_pot64 (uint64_t x)
|
884
|
-
ecb_function_ ecb_bool ecb_is_pot64 (uint64_t x) { return !(x & (x - 1)); }
|
990
|
+
ecb_function_ ecb_const ecb_bool ecb_is_pot32 (uint32_t x);
|
991
|
+
ecb_function_ ecb_const ecb_bool ecb_is_pot32 (uint32_t x) { return !(x & (x - 1)); }
|
992
|
+
ecb_function_ ecb_const ecb_bool ecb_is_pot64 (uint64_t x);
|
993
|
+
ecb_function_ ecb_const ecb_bool ecb_is_pot64 (uint64_t x) { return !(x & (x - 1)); }
|
885
994
|
|
886
|
-
ecb_function_ uint8_t ecb_bitrev8 (uint8_t x)
|
887
|
-
ecb_function_ uint8_t ecb_bitrev8 (uint8_t x)
|
995
|
+
ecb_function_ ecb_const uint8_t ecb_bitrev8 (uint8_t x);
|
996
|
+
ecb_function_ ecb_const uint8_t ecb_bitrev8 (uint8_t x)
|
888
997
|
{
|
889
998
|
return ( (x * 0x0802U & 0x22110U)
|
890
|
-
| (x * 0x8020U & 0x88440U)) * 0x10101U >> 16;
|
999
|
+
| (x * 0x8020U & 0x88440U)) * 0x10101U >> 16;
|
891
1000
|
}
|
892
1001
|
|
893
|
-
ecb_function_ uint16_t ecb_bitrev16 (uint16_t x)
|
894
|
-
ecb_function_ uint16_t ecb_bitrev16 (uint16_t x)
|
1002
|
+
ecb_function_ ecb_const uint16_t ecb_bitrev16 (uint16_t x);
|
1003
|
+
ecb_function_ ecb_const uint16_t ecb_bitrev16 (uint16_t x)
|
895
1004
|
{
|
896
1005
|
x = ((x >> 1) & 0x5555) | ((x & 0x5555) << 1);
|
897
1006
|
x = ((x >> 2) & 0x3333) | ((x & 0x3333) << 2);
|
@@ -901,8 +1010,8 @@ ecb_function_ uint16_t ecb_bitrev16 (uint16_t x)
|
|
901
1010
|
return x;
|
902
1011
|
}
|
903
1012
|
|
904
|
-
ecb_function_ uint32_t ecb_bitrev32 (uint32_t x)
|
905
|
-
ecb_function_ uint32_t ecb_bitrev32 (uint32_t x)
|
1013
|
+
ecb_function_ ecb_const uint32_t ecb_bitrev32 (uint32_t x);
|
1014
|
+
ecb_function_ ecb_const uint32_t ecb_bitrev32 (uint32_t x)
|
906
1015
|
{
|
907
1016
|
x = ((x >> 1) & 0x55555555) | ((x & 0x55555555) << 1);
|
908
1017
|
x = ((x >> 2) & 0x33333333) | ((x & 0x33333333) << 2);
|
@@ -915,71 +1024,80 @@ ecb_function_ uint32_t ecb_bitrev32 (uint32_t x)
|
|
915
1024
|
|
916
1025
|
/* popcount64 is only available on 64 bit cpus as gcc builtin */
|
917
1026
|
/* so for this version we are lazy */
|
918
|
-
ecb_function_ int ecb_popcount64 (uint64_t x)
|
919
|
-
ecb_function_ int
|
1027
|
+
ecb_function_ ecb_const int ecb_popcount64 (uint64_t x);
|
1028
|
+
ecb_function_ ecb_const int
|
920
1029
|
ecb_popcount64 (uint64_t x)
|
921
1030
|
{
|
922
1031
|
return ecb_popcount32 (x) + ecb_popcount32 (x >> 32);
|
923
1032
|
}
|
924
1033
|
|
925
|
-
ecb_inline uint8_t ecb_rotl8 (uint8_t x, unsigned int count)
|
926
|
-
ecb_inline uint8_t ecb_rotr8 (uint8_t x, unsigned int count)
|
927
|
-
ecb_inline uint16_t ecb_rotl16 (uint16_t x, unsigned int count)
|
928
|
-
ecb_inline uint16_t ecb_rotr16 (uint16_t x, unsigned int count)
|
929
|
-
ecb_inline uint32_t ecb_rotl32 (uint32_t x, unsigned int count)
|
930
|
-
ecb_inline uint32_t ecb_rotr32 (uint32_t x, unsigned int count)
|
931
|
-
ecb_inline uint64_t ecb_rotl64 (uint64_t x, unsigned int count)
|
932
|
-
ecb_inline uint64_t ecb_rotr64 (uint64_t x, unsigned int count)
|
933
|
-
|
934
|
-
ecb_inline uint8_t ecb_rotl8 (uint8_t x, unsigned int count) { return (x >> ( 8 - count)) | (x << count); }
|
935
|
-
ecb_inline uint8_t ecb_rotr8 (uint8_t x, unsigned int count) { return (x << ( 8 - count)) | (x >> count); }
|
936
|
-
ecb_inline uint16_t ecb_rotl16 (uint16_t x, unsigned int count) { return (x >> (16 - count)) | (x << count); }
|
937
|
-
ecb_inline uint16_t ecb_rotr16 (uint16_t x, unsigned int count) { return (x << (16 - count)) | (x >> count); }
|
938
|
-
ecb_inline uint32_t ecb_rotl32 (uint32_t x, unsigned int count) { return (x >> (32 - count)) | (x << count); }
|
939
|
-
ecb_inline uint32_t ecb_rotr32 (uint32_t x, unsigned int count) { return (x << (32 - count)) | (x >> count); }
|
940
|
-
ecb_inline uint64_t ecb_rotl64 (uint64_t x, unsigned int count) { return (x >> (64 - count)) | (x << count); }
|
941
|
-
ecb_inline uint64_t ecb_rotr64 (uint64_t x, unsigned int count) { return (x << (64 - count)) | (x >> count); }
|
942
|
-
|
943
|
-
#if ECB_GCC_VERSION(4,3)
|
1034
|
+
ecb_inline ecb_const uint8_t ecb_rotl8 (uint8_t x, unsigned int count);
|
1035
|
+
ecb_inline ecb_const uint8_t ecb_rotr8 (uint8_t x, unsigned int count);
|
1036
|
+
ecb_inline ecb_const uint16_t ecb_rotl16 (uint16_t x, unsigned int count);
|
1037
|
+
ecb_inline ecb_const uint16_t ecb_rotr16 (uint16_t x, unsigned int count);
|
1038
|
+
ecb_inline ecb_const uint32_t ecb_rotl32 (uint32_t x, unsigned int count);
|
1039
|
+
ecb_inline ecb_const uint32_t ecb_rotr32 (uint32_t x, unsigned int count);
|
1040
|
+
ecb_inline ecb_const uint64_t ecb_rotl64 (uint64_t x, unsigned int count);
|
1041
|
+
ecb_inline ecb_const uint64_t ecb_rotr64 (uint64_t x, unsigned int count);
|
1042
|
+
|
1043
|
+
ecb_inline ecb_const uint8_t ecb_rotl8 (uint8_t x, unsigned int count) { return (x >> ( 8 - count)) | (x << count); }
|
1044
|
+
ecb_inline ecb_const uint8_t ecb_rotr8 (uint8_t x, unsigned int count) { return (x << ( 8 - count)) | (x >> count); }
|
1045
|
+
ecb_inline ecb_const uint16_t ecb_rotl16 (uint16_t x, unsigned int count) { return (x >> (16 - count)) | (x << count); }
|
1046
|
+
ecb_inline ecb_const uint16_t ecb_rotr16 (uint16_t x, unsigned int count) { return (x << (16 - count)) | (x >> count); }
|
1047
|
+
ecb_inline ecb_const uint32_t ecb_rotl32 (uint32_t x, unsigned int count) { return (x >> (32 - count)) | (x << count); }
|
1048
|
+
ecb_inline ecb_const uint32_t ecb_rotr32 (uint32_t x, unsigned int count) { return (x << (32 - count)) | (x >> count); }
|
1049
|
+
ecb_inline ecb_const uint64_t ecb_rotl64 (uint64_t x, unsigned int count) { return (x >> (64 - count)) | (x << count); }
|
1050
|
+
ecb_inline ecb_const uint64_t ecb_rotr64 (uint64_t x, unsigned int count) { return (x << (64 - count)) | (x >> count); }
|
1051
|
+
|
1052
|
+
#if ECB_GCC_VERSION(4,3) || (ECB_CLANG_BUILTIN(__builtin_bswap32) && ECB_CLANG_BUILTIN(__builtin_bswap64))
|
1053
|
+
#if ECB_GCC_VERSION(4,8) || ECB_CLANG_BUILTIN(__builtin_bswap16)
|
1054
|
+
#define ecb_bswap16(x) __builtin_bswap16 (x)
|
1055
|
+
#else
|
944
1056
|
#define ecb_bswap16(x) (__builtin_bswap32 (x) >> 16)
|
1057
|
+
#endif
|
945
1058
|
#define ecb_bswap32(x) __builtin_bswap32 (x)
|
946
1059
|
#define ecb_bswap64(x) __builtin_bswap64 (x)
|
1060
|
+
#elif _MSC_VER
|
1061
|
+
#include <stdlib.h>
|
1062
|
+
#define ecb_bswap16(x) ((uint16_t)_byteswap_ushort ((uint16_t)(x)))
|
1063
|
+
#define ecb_bswap32(x) ((uint32_t)_byteswap_ulong ((uint32_t)(x)))
|
1064
|
+
#define ecb_bswap64(x) ((uint64_t)_byteswap_uint64 ((uint64_t)(x)))
|
947
1065
|
#else
|
948
|
-
ecb_function_ uint16_t ecb_bswap16 (uint16_t x)
|
949
|
-
ecb_function_ uint16_t
|
1066
|
+
ecb_function_ ecb_const uint16_t ecb_bswap16 (uint16_t x);
|
1067
|
+
ecb_function_ ecb_const uint16_t
|
950
1068
|
ecb_bswap16 (uint16_t x)
|
951
1069
|
{
|
952
1070
|
return ecb_rotl16 (x, 8);
|
953
1071
|
}
|
954
1072
|
|
955
|
-
ecb_function_ uint32_t ecb_bswap32 (uint32_t x)
|
956
|
-
ecb_function_ uint32_t
|
1073
|
+
ecb_function_ ecb_const uint32_t ecb_bswap32 (uint32_t x);
|
1074
|
+
ecb_function_ ecb_const uint32_t
|
957
1075
|
ecb_bswap32 (uint32_t x)
|
958
1076
|
{
|
959
1077
|
return (((uint32_t)ecb_bswap16 (x)) << 16) | ecb_bswap16 (x >> 16);
|
960
1078
|
}
|
961
1079
|
|
962
|
-
ecb_function_ uint64_t ecb_bswap64 (uint64_t x)
|
963
|
-
ecb_function_ uint64_t
|
1080
|
+
ecb_function_ ecb_const uint64_t ecb_bswap64 (uint64_t x);
|
1081
|
+
ecb_function_ ecb_const uint64_t
|
964
1082
|
ecb_bswap64 (uint64_t x)
|
965
1083
|
{
|
966
1084
|
return (((uint64_t)ecb_bswap32 (x)) << 32) | ecb_bswap32 (x >> 32);
|
967
1085
|
}
|
968
1086
|
#endif
|
969
1087
|
|
970
|
-
#if ECB_GCC_VERSION(4,5)
|
1088
|
+
#if ECB_GCC_VERSION(4,5) || ECB_CLANG_BUILTIN(__builtin_unreachable)
|
971
1089
|
#define ecb_unreachable() __builtin_unreachable ()
|
972
1090
|
#else
|
973
1091
|
/* this seems to work fine, but gcc always emits a warning for it :/ */
|
974
|
-
ecb_inline void ecb_unreachable (void)
|
975
|
-
ecb_inline void ecb_unreachable (void) { }
|
1092
|
+
ecb_inline ecb_noreturn void ecb_unreachable (void);
|
1093
|
+
ecb_inline ecb_noreturn void ecb_unreachable (void) { }
|
976
1094
|
#endif
|
977
1095
|
|
978
1096
|
/* try to tell the compiler that some condition is definitely true */
|
979
1097
|
#define ecb_assume(cond) if (!(cond)) ecb_unreachable (); else 0
|
980
1098
|
|
981
|
-
ecb_inline unsigned char ecb_byteorder_helper (void)
|
982
|
-
ecb_inline unsigned char
|
1099
|
+
ecb_inline ecb_const unsigned char ecb_byteorder_helper (void);
|
1100
|
+
ecb_inline ecb_const unsigned char
|
983
1101
|
ecb_byteorder_helper (void)
|
984
1102
|
{
|
985
1103
|
/* the union code still generates code under pressure in gcc, */
|
@@ -988,7 +1106,7 @@ ecb_byteorder_helper (void)
|
|
988
1106
|
/* the reason why we have this horrible preprocessor mess */
|
989
1107
|
/* is to avoid it in all cases, at least on common architectures */
|
990
1108
|
/* or when using a recent enough gcc version (>= 4.6) */
|
991
|
-
#if __i386 || __i386__
|
1109
|
+
#if ((__i386 || __i386__) && !__VOS__) || _M_IX86 || ECB_GCC_AMD64 || ECB_MSVC_AMD64
|
992
1110
|
return 0x44;
|
993
1111
|
#elif __BYTE_ORDER__ && __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
|
994
1112
|
return 0x44;
|
@@ -1004,10 +1122,10 @@ ecb_byteorder_helper (void)
|
|
1004
1122
|
#endif
|
1005
1123
|
}
|
1006
1124
|
|
1007
|
-
ecb_inline ecb_bool ecb_big_endian (void)
|
1008
|
-
ecb_inline ecb_bool ecb_big_endian (void) { return ecb_byteorder_helper () == 0x11; }
|
1009
|
-
ecb_inline ecb_bool ecb_little_endian (void)
|
1010
|
-
ecb_inline ecb_bool ecb_little_endian (void) { return ecb_byteorder_helper () == 0x44; }
|
1125
|
+
ecb_inline ecb_const ecb_bool ecb_big_endian (void);
|
1126
|
+
ecb_inline ecb_const ecb_bool ecb_big_endian (void) { return ecb_byteorder_helper () == 0x11; }
|
1127
|
+
ecb_inline ecb_const ecb_bool ecb_little_endian (void);
|
1128
|
+
ecb_inline ecb_const ecb_bool ecb_little_endian (void) { return ecb_byteorder_helper () == 0x44; }
|
1011
1129
|
|
1012
1130
|
#if ECB_GCC_VERSION(3,0) || ECB_C99
|
1013
1131
|
#define ecb_mod(m,n) ((m) % (n) + ((m) % (n) < 0 ? (n) : 0))
|
@@ -1015,7 +1133,7 @@ ecb_inline ecb_bool ecb_little_endian (void) { return ecb_byteorder_helper () ==
|
|
1015
1133
|
#define ecb_mod(m,n) ((m) < 0 ? ((n) - 1 - ((-1 - (m)) % (n))) : ((m) % (n)))
|
1016
1134
|
#endif
|
1017
1135
|
|
1018
|
-
#if
|
1136
|
+
#if ECB_CPP
|
1019
1137
|
template<typename T>
|
1020
1138
|
static inline T ecb_div_rd (T val, T div)
|
1021
1139
|
{
|
@@ -1049,27 +1167,70 @@ ecb_inline ecb_bool ecb_little_endian (void) { return ecb_byteorder_helper () ==
|
|
1049
1167
|
/* the only noteworthy exception is ancient armle, which uses order 43218765 */
|
1050
1168
|
#if 0 \
|
1051
1169
|
|| __i386 || __i386__ \
|
1052
|
-
||
|
1170
|
+
|| ECB_GCC_AMD64 \
|
1053
1171
|
|| __powerpc__ || __ppc__ || __powerpc64__ || __ppc64__ \
|
1054
|
-
|| defined __arm__ && defined __ARM_EABI__ \
|
1055
1172
|
|| defined __s390__ || defined __s390x__ \
|
1056
1173
|
|| defined __mips__ \
|
1057
1174
|
|| defined __alpha__ \
|
1058
1175
|
|| defined __hppa__ \
|
1059
1176
|
|| defined __ia64__ \
|
1060
|
-
|| defined
|
1177
|
+
|| defined __m68k__ \
|
1178
|
+
|| defined __m88k__ \
|
1179
|
+
|| defined __sh__ \
|
1180
|
+
|| defined _M_IX86 || defined ECB_MSVC_AMD64 || defined _M_IA64 \
|
1181
|
+
|| (defined __arm__ && (defined __ARM_EABI__ || defined __EABI__ || defined __VFP_FP__ || defined _WIN32_WCE || defined __ANDROID__)) \
|
1182
|
+
|| defined __aarch64__
|
1061
1183
|
#define ECB_STDFP 1
|
1062
1184
|
#include <string.h> /* for memcpy */
|
1063
1185
|
#else
|
1064
1186
|
#define ECB_STDFP 0
|
1065
|
-
#include <math.h> /* for frexp*, ldexp* */
|
1066
1187
|
#endif
|
1067
1188
|
|
1068
1189
|
#ifndef ECB_NO_LIBM
|
1069
1190
|
|
1191
|
+
#include <math.h> /* for frexp*, ldexp*, INFINITY, NAN */
|
1192
|
+
|
1193
|
+
/* only the oldest of old doesn't have this one. solaris. */
|
1194
|
+
#ifdef INFINITY
|
1195
|
+
#define ECB_INFINITY INFINITY
|
1196
|
+
#else
|
1197
|
+
#define ECB_INFINITY HUGE_VAL
|
1198
|
+
#endif
|
1199
|
+
|
1200
|
+
#ifdef NAN
|
1201
|
+
#define ECB_NAN NAN
|
1202
|
+
#else
|
1203
|
+
#define ECB_NAN ECB_INFINITY
|
1204
|
+
#endif
|
1205
|
+
|
1206
|
+
#if ECB_C99 || _XOPEN_VERSION >= 600 || _POSIX_VERSION >= 200112L
|
1207
|
+
#define ecb_ldexpf(x,e) ldexpf ((x), (e))
|
1208
|
+
#define ecb_frexpf(x,e) frexpf ((x), (e))
|
1209
|
+
#else
|
1210
|
+
#define ecb_ldexpf(x,e) (float) ldexp ((double) (x), (e))
|
1211
|
+
#define ecb_frexpf(x,e) (float) frexp ((double) (x), (e))
|
1212
|
+
#endif
|
1213
|
+
|
1214
|
+
/* converts an ieee half/binary16 to a float */
|
1215
|
+
ecb_function_ ecb_const float ecb_binary16_to_float (uint16_t x);
|
1216
|
+
ecb_function_ ecb_const float
|
1217
|
+
ecb_binary16_to_float (uint16_t x)
|
1218
|
+
{
|
1219
|
+
int e = (x >> 10) & 0x1f;
|
1220
|
+
int m = x & 0x3ff;
|
1221
|
+
float r;
|
1222
|
+
|
1223
|
+
if (!e ) r = ecb_ldexpf (m , -24);
|
1224
|
+
else if (e != 31) r = ecb_ldexpf (m + 0x400, e - 25);
|
1225
|
+
else if (m ) r = ECB_NAN;
|
1226
|
+
else r = ECB_INFINITY;
|
1227
|
+
|
1228
|
+
return x & 0x8000 ? -r : r;
|
1229
|
+
}
|
1230
|
+
|
1070
1231
|
/* convert a float to ieee single/binary32 */
|
1071
|
-
ecb_function_ uint32_t ecb_float_to_binary32 (float x)
|
1072
|
-
ecb_function_ uint32_t
|
1232
|
+
ecb_function_ ecb_const uint32_t ecb_float_to_binary32 (float x);
|
1233
|
+
ecb_function_ ecb_const uint32_t
|
1073
1234
|
ecb_float_to_binary32 (float x)
|
1074
1235
|
{
|
1075
1236
|
uint32_t r;
|
@@ -1086,7 +1247,7 @@ ecb_inline ecb_bool ecb_little_endian (void) { return ecb_byteorder_helper () ==
|
|
1086
1247
|
if (x < -3.40282346638528860e+38f) return 0xff800000U;
|
1087
1248
|
if (x != x ) return 0x7fbfffffU;
|
1088
1249
|
|
1089
|
-
m =
|
1250
|
+
m = ecb_frexpf (x, &e) * 0x1000000U;
|
1090
1251
|
|
1091
1252
|
r = m & 0x80000000U;
|
1092
1253
|
|
@@ -1108,8 +1269,8 @@ ecb_inline ecb_bool ecb_little_endian (void) { return ecb_byteorder_helper () ==
|
|
1108
1269
|
}
|
1109
1270
|
|
1110
1271
|
/* converts an ieee single/binary32 to a float */
|
1111
|
-
ecb_function_ float ecb_binary32_to_float (uint32_t x)
|
1112
|
-
ecb_function_ float
|
1272
|
+
ecb_function_ ecb_const float ecb_binary32_to_float (uint32_t x);
|
1273
|
+
ecb_function_ ecb_const float
|
1113
1274
|
ecb_binary32_to_float (uint32_t x)
|
1114
1275
|
{
|
1115
1276
|
float r;
|
@@ -1129,7 +1290,7 @@ ecb_inline ecb_bool ecb_little_endian (void) { return ecb_byteorder_helper () ==
|
|
1129
1290
|
e = 1;
|
1130
1291
|
|
1131
1292
|
/* we distrust ldexpf a bit and do the 2**-24 scaling by an extra multiply */
|
1132
|
-
r =
|
1293
|
+
r = ecb_ldexpf (x * (0.5f / 0x800000U), e - 126);
|
1133
1294
|
|
1134
1295
|
r = neg ? -r : r;
|
1135
1296
|
#endif
|
@@ -1138,8 +1299,8 @@ ecb_inline ecb_bool ecb_little_endian (void) { return ecb_byteorder_helper () ==
|
|
1138
1299
|
}
|
1139
1300
|
|
1140
1301
|
/* convert a double to ieee double/binary64 */
|
1141
|
-
ecb_function_ uint64_t ecb_double_to_binary64 (double x)
|
1142
|
-
ecb_function_ uint64_t
|
1302
|
+
ecb_function_ ecb_const uint64_t ecb_double_to_binary64 (double x);
|
1303
|
+
ecb_function_ ecb_const uint64_t
|
1143
1304
|
ecb_double_to_binary64 (double x)
|
1144
1305
|
{
|
1145
1306
|
uint64_t r;
|
@@ -1178,8 +1339,8 @@ ecb_inline ecb_bool ecb_little_endian (void) { return ecb_byteorder_helper () ==
|
|
1178
1339
|
}
|
1179
1340
|
|
1180
1341
|
/* converts an ieee double/binary64 to a double */
|
1181
|
-
ecb_function_ double ecb_binary64_to_double (uint64_t x)
|
1182
|
-
ecb_function_ double
|
1342
|
+
ecb_function_ ecb_const double ecb_binary64_to_double (uint64_t x);
|
1343
|
+
ecb_function_ ecb_const double
|
1183
1344
|
ecb_binary64_to_double (uint64_t x)
|
1184
1345
|
{
|
1185
1346
|
double r;
|
@@ -2017,7 +2178,7 @@ downheap (ANHE *heap, int N, int k)
|
|
2017
2178
|
|
2018
2179
|
heap [k] = heap [c];
|
2019
2180
|
ev_active (ANHE_w (heap [k])) = k;
|
2020
|
-
|
2181
|
+
|
2021
2182
|
k = c;
|
2022
2183
|
}
|
2023
2184
|
|
@@ -2110,8 +2271,6 @@ evpipe_init (EV_P)
|
|
2110
2271
|
fd_intern (fds [0]);
|
2111
2272
|
}
|
2112
2273
|
|
2113
|
-
fd_intern (fds [1]);
|
2114
|
-
|
2115
2274
|
evpipe [0] = fds [0];
|
2116
2275
|
|
2117
2276
|
if (evpipe [1] < 0)
|
@@ -2127,6 +2286,8 @@ evpipe_init (EV_P)
|
|
2127
2286
|
close (fds [1]);
|
2128
2287
|
}
|
2129
2288
|
|
2289
|
+
fd_intern (evpipe [1]);
|
2290
|
+
|
2130
2291
|
ev_io_set (&pipe_w, evpipe [0] < 0 ? evpipe [1] : evpipe [0], EV_READ);
|
2131
2292
|
ev_io_start (EV_A_ &pipe_w);
|
2132
2293
|
ev_unref (EV_A); /* watcher should not keep loop alive */
|
@@ -2437,7 +2598,7 @@ ev_supported_backends (void) EV_THROW
|
|
2437
2598
|
if (EV_USE_EPOLL ) flags |= EVBACKEND_EPOLL;
|
2438
2599
|
if (EV_USE_POLL ) flags |= EVBACKEND_POLL;
|
2439
2600
|
if (EV_USE_SELECT) flags |= EVBACKEND_SELECT;
|
2440
|
-
|
2601
|
+
|
2441
2602
|
return flags;
|
2442
2603
|
}
|
2443
2604
|
|
@@ -2519,7 +2680,7 @@ ev_userdata (EV_P) EV_THROW
|
|
2519
2680
|
}
|
2520
2681
|
|
2521
2682
|
void
|
2522
|
-
ev_set_invoke_pending_cb (EV_P_
|
2683
|
+
ev_set_invoke_pending_cb (EV_P_ ev_loop_callback invoke_pending_cb) EV_THROW
|
2523
2684
|
{
|
2524
2685
|
invoke_cb = invoke_pending_cb;
|
2525
2686
|
}
|
@@ -3243,12 +3404,18 @@ time_update (EV_P_ ev_tstamp max_block)
|
|
3243
3404
|
|
3244
3405
|
/* ########## COOLIO PATCHERY HO! ########## */
|
3245
3406
|
#if defined(HAVE_RB_THREAD_BLOCKING_REGION) || defined(HAVE_RB_THREAD_CALL_WITHOUT_GVL)
|
3407
|
+
struct ev_poll_args {
|
3408
|
+
struct ev_loop *loop;
|
3409
|
+
ev_tstamp waittime;
|
3410
|
+
};
|
3411
|
+
|
3246
3412
|
static
|
3247
|
-
VALUE ev_backend_poll(void
|
3413
|
+
VALUE ev_backend_poll(void *ptr)
|
3248
3414
|
{
|
3249
|
-
struct
|
3250
|
-
|
3251
|
-
backend_poll (EV_A_ waittime);
|
3415
|
+
struct ev_poll_args *args = (struct ev_poll_args *)ptr;
|
3416
|
+
struct ev_loop *loop = args->loop;
|
3417
|
+
backend_poll (EV_A_ args->waittime);
|
3418
|
+
return Qnil;
|
3252
3419
|
}
|
3253
3420
|
#endif
|
3254
3421
|
/* ######################################## */
|
@@ -3258,7 +3425,7 @@ ev_run (EV_P_ int flags)
|
|
3258
3425
|
{
|
3259
3426
|
/* ########## COOLIO PATCHERY HO! ########## */
|
3260
3427
|
#if defined(HAVE_RB_THREAD_BLOCKING_REGION) || defined(HAVE_RB_THREAD_CALL_WITHOUT_GVL)
|
3261
|
-
|
3428
|
+
struct ev_poll_args poll_args;
|
3262
3429
|
#endif
|
3263
3430
|
/* ######################################## */
|
3264
3431
|
|
@@ -3383,9 +3550,6 @@ ev_run (EV_P_ int flags)
|
|
3383
3550
|
/*
|
3384
3551
|
########################## COOLIO PATCHERY HO! ##########################
|
3385
3552
|
|
3386
|
-
The original patch file is made by Tony Arcieri.
|
3387
|
-
https://github.com/celluloid/nio4r/blob/680143345726c5a64bb22376ca8fc3c6857019ae/ext/libev/ruby_gil.patch.
|
3388
|
-
|
3389
3553
|
According to the grandwizards of Ruby, locking and unlocking of the global
|
3390
3554
|
interpreter lock are apparently too powerful a concept for a mere mortal to
|
3391
3555
|
wield (although redefining what + and - do to numbers is totally cool).
|
@@ -3410,7 +3574,15 @@ other and thus cannot be composed.
|
|
3410
3574
|
And thus we are left with no choice but to patch the internals of libev in
|
3411
3575
|
order to release a mutex at just the precise moment.
|
3412
3576
|
|
3413
|
-
|
3577
|
+
This is a great example of a situation where granular locking and unlocking
|
3578
|
+
of the GVL is practically required. The goal is to get as close to the
|
3579
|
+
system call as possible, and to keep the GVL unlocked for the shortest
|
3580
|
+
amount of time possible.
|
3581
|
+
|
3582
|
+
Perhaps Ruby could benefit from such an API, e.g:
|
3583
|
+
|
3584
|
+
rb_thread_unsafe_dangerous_crazy_blocking_region_begin(...);
|
3585
|
+
rb_thread_unsafe_dangerous_crazy_blocking_region_end(...);
|
3414
3586
|
|
3415
3587
|
#######################################################################
|
3416
3588
|
*/
|
@@ -3422,19 +3594,17 @@ Let this be a lesson to the all: CALLBACKS FUCKING BLOW
|
|
3422
3594
|
|
3423
3595
|
#ifndef HAVE_RB_THREAD_CALL_WITHOUT_GVL
|
3424
3596
|
#ifdef HAVE_RB_THREAD_BLOCKING_REGION
|
3425
|
-
|
3426
3597
|
#define rb_thread_call_without_gvl(func, data1, ubf, data2) \
|
3427
3598
|
rb_thread_blocking_region((rb_blocking_function_t *)func, data1, ubf, data2)
|
3428
|
-
|
3429
3599
|
#endif
|
3430
3600
|
#endif
|
3431
3601
|
|
3432
3602
|
#if defined(HAVE_RB_THREAD_BLOCKING_REGION) || defined(HAVE_RB_THREAD_CALL_WITHOUT_GVL)
|
3433
|
-
poll_args
|
3434
|
-
poll_args
|
3603
|
+
poll_args.loop = loop;
|
3604
|
+
poll_args.waittime = waittime;
|
3435
3605
|
rb_thread_call_without_gvl(ev_backend_poll, (void *)&poll_args, RUBY_UBF_IO, 0);
|
3436
3606
|
#else
|
3437
|
-
|
3607
|
+
backend_poll (EV_A_ waittime);
|
3438
3608
|
#endif
|
3439
3609
|
/*
|
3440
3610
|
############################# END PATCHERY ############################
|
@@ -4882,4 +5052,3 @@ ev_walk (EV_P_ int types, void (*cb)(EV_P_ int type, void *w)) EV_THROW
|
|
4882
5052
|
#if EV_MULTIPLICITY
|
4883
5053
|
#include "ev_wrap.h"
|
4884
5054
|
#endif
|
4885
|
-
|