Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
pygolang
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
pygolang
Commits
2b56ce71
Commit
2b56ce71
authored
Apr 30, 2023
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Plain Diff
Sync with master
parents
300d7dfa
55d39d4d
Changes
39
Hide whitespace changes
Inline
Side-by-side
Showing
39 changed files
with
770 additions
and
284 deletions
+770
-284
.gitignore
.gitignore
+2
-0
MANIFEST.in
MANIFEST.in
+4
-1
golang/__init__.py
golang/__init__.py
+4
-1
golang/_compat/windows/strings.h
golang/_compat/windows/strings.h
+30
-0
golang/_compat/windows/unistd.h
golang/_compat/windows/unistd.h
+31
-0
golang/_golang.pyx
golang/_golang.pyx
+2
-2
golang/fmt.cpp
golang/fmt.cpp
+3
-3
golang/fmt.h
golang/fmt.h
+7
-4
golang/golang_test.py
golang/golang_test.py
+79
-16
golang/libgolang.h
golang/libgolang.h
+9
-3
golang/os.cpp
golang/os.cpp
+18
-5
golang/os.h
golang/os.h
+10
-4
golang/os/signal.cpp
golang/os/signal.cpp
+113
-15
golang/os/signal_test.py
golang/os/signal_test.py
+80
-64
golang/os/testprog/signal_test_all.py
golang/os/testprog/signal_test_all.py
+14
-12
golang/pyx/build.py
golang/pyx/build.py
+54
-13
golang/pyx/runtime.h
golang/pyx/runtime.h
+3
-3
golang/pyx/testprog/cmdclass_custom.py
golang/pyx/testprog/cmdclass_custom.py
+7
-2
golang/pyx/testprog/golang_dso_user/dsouser/dso.cpp
golang/pyx/testprog/golang_dso_user/dsouser/dso.cpp
+3
-3
golang/pyx/testprog/golang_dso_user/dsouser/dso.h
golang/pyx/testprog/golang_dso_user/dsouser/dso.h
+33
-0
golang/pyx/testprog/golang_dso_user/dsouser/test.pyx
golang/pyx/testprog/golang_dso_user/dsouser/test.pyx
+3
-6
golang/pyx/testprog/golang_dso_user/setup.py
golang/pyx/testprog/golang_dso_user/setup.py
+5
-3
golang/runtime/_runtime_gevent.pyx
golang/runtime/_runtime_gevent.pyx
+54
-28
golang/runtime/internal/atomic.cpp
golang/runtime/internal/atomic.cpp
+7
-2
golang/runtime/internal/syscall.cpp
golang/runtime/internal/syscall.cpp
+51
-7
golang/runtime/internal/syscall.h
golang/runtime/internal/syscall.h
+16
-10
golang/runtime/libgolang.cpp
golang/runtime/libgolang.cpp
+20
-2
golang/sync.h
golang/sync.h
+2
-2
golang/testprog/golang_test_defer_excchain.txt
golang/testprog/golang_test_defer_excchain.txt
+4
-1
golang/testprog/golang_test_defer_excchain.txt-pytest
golang/testprog/golang_test_defer_excchain.txt-pytest
+1
-1
golang/testprog/golang_test_str.py
golang/testprog/golang_test_str.py
+4
-4
golang/testprog/golang_test_str.txt
golang/testprog/golang_test_str.txt
+7
-7
golang/time.h
golang/time.h
+2
-2
gpython/__init__.py
gpython/__init__.py
+29
-16
gpython/gpython_test.py
gpython/gpython_test.py
+5
-5
gpython/testprog/print_opt.py
gpython/testprog/print_opt.py
+8
-7
pyproject.toml
pyproject.toml
+1
-1
setup.py
setup.py
+36
-23
tox.ini
tox.ini
+9
-6
No files found.
.gitignore
View file @
2b56ce71
...
@@ -10,6 +10,8 @@ build/
...
@@ -10,6 +10,8 @@ build/
*.so.*
*.so.*
*.dylib
*.dylib
*.dll
*.dll
*.lib
*.exp
*.pyd
*.pyd
*_dsoinfo.py
*_dsoinfo.py
...
...
MANIFEST.in
View file @
2b56ce71
...
@@ -3,7 +3,7 @@ include golang/libgolang.h
...
@@ -3,7 +3,7 @@ include golang/libgolang.h
include golang/runtime/libgolang.cpp
include golang/runtime/libgolang.cpp
include golang/runtime/libpyxruntime.cpp
include golang/runtime/libpyxruntime.cpp
include golang/pyx/runtime.h
include golang/pyx/runtime.h
include golang/pyx/
runtime.cpp
include golang/pyx/
testprog/golang_dso_user/dsouser/dso.h
include golang/pyx/testprog/golang_dso_user/dsouser/dso.cpp
include golang/pyx/testprog/golang_dso_user/dsouser/dso.cpp
include golang/runtime/internal.h
include golang/runtime/internal.h
include golang/runtime/internal/atomic.h
include golang/runtime/internal/atomic.h
...
@@ -34,6 +34,9 @@ include golang/sync_test.cpp
...
@@ -34,6 +34,9 @@ include golang/sync_test.cpp
include golang/time.h
include golang/time.h
include golang/time.cpp
include golang/time.cpp
include golang/_testing.h
include golang/_testing.h
include golang/_compat/windows/strings.h
include golang/_compat/windows/unistd.h
recursive-include golang *.py *.pxd *.pyx *.toml *.txt*
recursive-include golang *.py *.pxd *.pyx *.toml *.txt*
recursive-include gpython *.py
recursive-include gpython *.py
recursive-include 3rdparty *.h
recursive-include 3rdparty *.h
recursive-exclude golang *_dsoinfo.py
golang/__init__.py
View file @
2b56ce71
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
# Copyright (C) 2018-202
2
Nexedi SA and Contributors.
# Copyright (C) 2018-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -42,6 +42,9 @@ from golang._gopath import gimport # make gimport available from golang
...
@@ -42,6 +42,9 @@ from golang._gopath import gimport # make gimport available from golang
import
inspect
,
sys
import
inspect
,
sys
import
decorator
,
six
import
decorator
,
six
import
setuptools_dso
setuptools_dso
.
dylink_prepare_dso
(
'golang.runtime.libgolang'
)
from
golang._golang
import
_pysys_exc_clear
as
_sys_exc_clear
from
golang._golang
import
_pysys_exc_clear
as
_sys_exc_clear
# @func is a necessary decorator for functions for selected golang features to work.
# @func is a necessary decorator for functions for selected golang features to work.
...
...
golang/_compat/windows/strings.h
0 → 100644
View file @
2b56ce71
#ifndef _NXD_LIBGOLANG_COMPAT_WIN_STRINGS_H
#define _NXD_LIBGOLANG_COMPAT_WIN_STRINGS_H
//
// Copyright (C) 2023 Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
//
// This program is free software: you can Use, Study, Modify and Redistribute
// it under the terms of the GNU General Public License version 3, or (at your
// option) any later version, as published by the Free Software Foundation.
//
// You can also Link and Combine this program with other software covered by
// the terms of any of the Free Software licenses or any of the Open Source
// Initiative approved licenses and Convey the resulting work. Corresponding
// source of such a combination shall include the source code for all other
// software used.
//
// This program is distributed WITHOUT ANY WARRANTY; without even the implied
// warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
//
// See COPYING file for full licensing terms.
// See https://www.nexedi.com/licensing for rationale and options.
#include <string.h>
static
inline
void
bzero
(
void
*
p
,
size_t
n
)
{
memset
(
p
,
'\0'
,
n
);
}
#endif // _NXD_LIBGOLANG_COMPAT_WIN_STRINGS_H
golang/_compat/windows/unistd.h
0 → 100644
View file @
2b56ce71
#ifndef _NXD_LIBGOLANG_COMPAT_WIN_UNISTD_H
#define _NXD_LIBGOLANG_COMPAT_WIN_UNISTD_H
//
// Copyright (C) 2023 Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
//
// This program is free software: you can Use, Study, Modify and Redistribute
// it under the terms of the GNU General Public License version 3, or (at your
// option) any later version, as published by the Free Software Foundation.
//
// You can also Link and Combine this program with other software covered by
// the terms of any of the Free Software licenses or any of the Open Source
// Initiative approved licenses and Convey the resulting work. Corresponding
// source of such a combination shall include the source code for all other
// software used.
//
// This program is distributed WITHOUT ANY WARRANTY; without even the implied
// warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
//
// See COPYING file for full licensing terms.
// See https://www.nexedi.com/licensing for rationale and options.
// stub unistd.h to be used on Windows where it is absent.
// we need this because e.g. `cimport posix.stat` forces inclusion of unistd.h
// even if we use part of posix.stat that is available everywhere.
#include <io.h>
#define O_CLOEXEC _O_NOINHERIT
#endif // _NXD_LIBGOLANG_COMPAT_WIN_UNISTD_H
golang/_golang.pyx
View file @
2b56ce71
...
@@ -5,7 +5,7 @@
...
@@ -5,7 +5,7 @@
# distutils: language = c++
# distutils: language = c++
# distutils: depends = libgolang.h os/signal.h _golang_str.pyx
# distutils: depends = libgolang.h os/signal.h _golang_str.pyx
#
#
# Copyright (C) 2018-202
2
Nexedi SA and Contributors.
# Copyright (C) 2018-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -469,7 +469,7 @@ def pyselect(*pycasev):
...
@@ -469,7 +469,7 @@ def pyselect(*pycasev):
casev
[
i
].
user
=
pych
.
dtype
casev
[
i
].
user
=
pych
.
dtype
with
nogil
:
with
nogil
:
selected
=
_chanselect_pyexc
(
&
casev
[
0
]
,
casev
.
size
())
selected
=
_chanselect_pyexc
(
casev
.
data
()
,
casev
.
size
())
finally
:
finally
:
# decref not sent tx (see ^^^ send prepare)
# decref not sent tx (see ^^^ send prepare)
...
...
golang/fmt.cpp
View file @
2b56ce71
// Copyright (C) 2019-202
0
Nexedi SA and Contributors.
// Copyright (C) 2019-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -48,7 +48,7 @@ string _vsprintf(const char *format, va_list argp) {
...
@@ -48,7 +48,7 @@ string _vsprintf(const char *format, va_list argp) {
return
string
(
buf
.
get
(),
buf
.
get
()
+
nchar
);
// without trailing '\0'
return
string
(
buf
.
get
(),
buf
.
get
()
+
nchar
);
// without trailing '\0'
}
}
string
sprintf
(
const
string
&
format
,
...)
{
string
sprintf
(
const
string
format
,
...)
{
va_list
argp
;
va_list
argp
;
va_start
(
argp
,
format
);
va_start
(
argp
,
format
);
string
str
=
fmt
::
_vsprintf
(
format
.
c_str
(),
argp
);
string
str
=
fmt
::
_vsprintf
(
format
.
c_str
(),
argp
);
...
@@ -64,7 +64,7 @@ string sprintf(const char *format, ...) {
...
@@ -64,7 +64,7 @@ string sprintf(const char *format, ...) {
return
str
;
return
str
;
}
}
error
___errorf
(
const
string
&
format
,
...)
{
error
___errorf
(
const
string
format
,
...)
{
va_list
argp
;
va_list
argp
;
va_start
(
argp
,
format
);
va_start
(
argp
,
format
);
error
err
=
errors
::
New
(
fmt
::
_vsprintf
(
format
.
c_str
(),
argp
));
error
err
=
errors
::
New
(
fmt
::
_vsprintf
(
format
.
c_str
(),
argp
));
...
...
golang/fmt.h
View file @
2b56ce71
#ifndef _NXD_LIBGOLANG_FMT_H
#ifndef _NXD_LIBGOLANG_FMT_H
#define _NXD_LIBGOLANG_FMT_H
#define _NXD_LIBGOLANG_FMT_H
// Copyright (C) 2019-202
0
Nexedi SA and Contributors.
// Copyright (C) 2019-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -40,7 +40,7 @@ namespace golang {
...
@@ -40,7 +40,7 @@ namespace golang {
namespace
fmt
{
namespace
fmt
{
// sprintf formats text into string.
// sprintf formats text into string.
LIBGOLANG_API
string
sprintf
(
const
string
&
format
,
...);
LIBGOLANG_API
string
sprintf
(
const
string
format
,
...);
// intseq<i1, i2, ...> and intrange<n> are used by errorf to handle %w.
// intseq<i1, i2, ...> and intrange<n> are used by errorf to handle %w.
...
@@ -75,7 +75,7 @@ namespace {
...
@@ -75,7 +75,7 @@ namespace {
//
//
// format suffix ": %w" is additionally handled as in Go with
// format suffix ": %w" is additionally handled as in Go with
// `errorf("... : %w", ..., err)` creating error that can be unwrapped back to err.
// `errorf("... : %w", ..., err)` creating error that can be unwrapped back to err.
LIBGOLANG_API
error
___errorf
(
const
string
&
format
,
...);
LIBGOLANG_API
error
___errorf
(
const
string
format
,
...);
LIBGOLANG_API
error
___errorfTryWrap
(
const
string
&
format
,
error
last_err
,
...);
LIBGOLANG_API
error
___errorfTryWrap
(
const
string
&
format
,
error
last_err
,
...);
LIBGOLANG_API
string
___error_str
(
error
err
);
LIBGOLANG_API
string
___error_str
(
error
err
);
...
@@ -111,7 +111,10 @@ inline error errorf(const string& format, Argv... argv) {
...
@@ -111,7 +111,10 @@ inline error errorf(const string& format, Argv... argv) {
// `const char *` overloads just to catch format mistakes as
// `const char *` overloads just to catch format mistakes as
// __attribute__(format) does not work with std::string.
// __attribute__(format) does not work with std::string.
LIBGOLANG_API
string
sprintf
(
const
char
*
format
,
...)
LIBGOLANG_API
string
sprintf
(
const
char
*
format
,
...)
__attribute__
((
format
(
printf
,
1
,
2
)));
#ifndef _MSC_VER
__attribute__
((
format
(
printf
,
1
,
2
)))
#endif
;
// cannot use __attribute__(format) for errorf as we add %w handling.
// cannot use __attribute__(format) for errorf as we add %w handling.
// still `const char *` overload is useful for performance.
// still `const char *` overload is useful for performance.
...
...
golang/golang_test.py
View file @
2b56ce71
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
# Copyright (C) 2018-202
2
Nexedi SA and Contributors.
# Copyright (C) 2018-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -23,14 +23,14 @@ from __future__ import print_function, absolute_import
...
@@ -23,14 +23,14 @@ from __future__ import print_function, absolute_import
from
golang
import
go
,
chan
,
select
,
default
,
nilchan
,
_PanicError
,
func
,
panic
,
\
from
golang
import
go
,
chan
,
select
,
default
,
nilchan
,
_PanicError
,
func
,
panic
,
\
defer
,
recover
,
u
defer
,
recover
,
u
from
golang
import
sync
from
golang
import
sync
from
pytest
import
raises
,
mark
,
fail
from
pytest
import
raises
,
mark
,
fail
,
skip
from
_pytest._code
import
Traceback
from
_pytest._code
import
Traceback
from
os.path
import
dirname
from
os.path
import
dirname
import
os
,
sys
,
inspect
,
importlib
,
traceback
,
doctest
import
os
,
sys
,
inspect
,
importlib
,
traceback
,
doctest
from
subprocess
import
Popen
,
PIPE
from
subprocess
import
Popen
,
PIPE
import
six
import
six
from
six.moves
import
range
as
xrange
from
six.moves
import
range
as
xrange
import
gc
,
weakref
,
warnings
import
gc
,
weakref
import
re
import
re
from
golang
import
_golang_test
from
golang
import
_golang_test
...
@@ -1559,6 +1559,7 @@ RuntimeError: gamma
...
@@ -1559,6 +1559,7 @@ RuntimeError: gamma
Traceback (most recent call last):
Traceback (most recent call last):
File "PYGOLANG/golang/__init__.py", line ..., in _
File "PYGOLANG/golang/__init__.py", line ..., in _
return f(*argv, **kw)
return f(*argv, **kw)
^^^^^^^^^^^^^^ +PY311
File "PYGOLANG/golang/golang_test.py", line ..., in caller
File "PYGOLANG/golang/golang_test.py", line ..., in caller
raise RuntimeError("ccc")
raise RuntimeError("ccc")
RuntimeError: ccc
RuntimeError: ccc
...
@@ -1579,9 +1580,11 @@ Traceback (most recent call last):
...
@@ -1579,9 +1580,11 @@ Traceback (most recent call last):
caller()
caller()
...
...
File "PYGOLANG/golang/__init__.py", line ..., in _
File "PYGOLANG/golang/__init__.py", line ..., in _
return f(*argv, **kw)
return f(*argv, **kw) -PY310
with __goframe__: +PY310
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
d()
d() -PY310
with __goframe__: +PY310
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
d()
d()
File "PYGOLANG/golang/golang_test.py", line ..., in q1
File "PYGOLANG/golang/golang_test.py", line ..., in q1
...
@@ -1596,9 +1599,11 @@ Traceback (most recent call last):
...
@@ -1596,9 +1599,11 @@ Traceback (most recent call last):
caller()
caller()
...
...
File "PYGOLANG/golang/__init__.py", line ..., in _
File "PYGOLANG/golang/__init__.py", line ..., in _
return f(*argv, **kw)
return f(*argv, **kw) -PY310
with __goframe__: +PY310
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
d()
d() -PY310
with __goframe__: +PY310
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
d()
d()
File "PYGOLANG/golang/golang_test.py", line ..., in q1
File "PYGOLANG/golang/golang_test.py", line ..., in q1
...
@@ -1611,6 +1616,7 @@ RuntimeError: aaa
...
@@ -1611,6 +1616,7 @@ RuntimeError: aaa
Traceback (most recent call last):
Traceback (most recent call last):
File "PYGOLANG/golang/__init__.py", line ..., in _
File "PYGOLANG/golang/__init__.py", line ..., in _
return f(*argv, **kw)
return f(*argv, **kw)
^^^^^^^^^^^^^^ +PY311
File "PYGOLANG/golang/golang_test.py", line ..., in caller
File "PYGOLANG/golang/golang_test.py", line ..., in caller
raise RuntimeError("ccc")
raise RuntimeError("ccc")
RuntimeError: ccc
RuntimeError: ccc
...
@@ -1631,9 +1637,11 @@ Traceback (most recent call last):
...
@@ -1631,9 +1637,11 @@ Traceback (most recent call last):
caller()
caller()
...
...
File "PYGOLANG/golang/__init__.py", line ..., in _
File "PYGOLANG/golang/__init__.py", line ..., in _
return f(*argv, **kw)
return f(*argv, **kw) -PY310
with __goframe__: +PY310
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
d()
d() -PY310
with __goframe__: +PY310
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
d()
d()
File "PYGOLANG/golang/golang_test.py", line ..., in q1
File "PYGOLANG/golang/golang_test.py", line ..., in q1
...
@@ -1654,6 +1662,13 @@ def test_defer_excchain_dump():
...
@@ -1654,6 +1662,13 @@ def test_defer_excchain_dump():
# ----//---- (ipython)
# ----//---- (ipython)
def
test_defer_excchain_dump_ipython
():
def
test_defer_excchain_dump_ipython
():
# ipython 8 changed traceback output significantly
# we do not need to test it because we acticate ipython-related patch only
# on py2 for which latest ipython version is 5.
import
IPython
if
six
.
PY3
and
IPython
.
version_info
>=
(
8
,
0
):
skip
(
"ipython is patched only on py2; ipython8 changed traceback format"
)
tbok
=
readfile
(
dir_testprog
+
"/golang_test_defer_excchain.txt-ipython"
)
tbok
=
readfile
(
dir_testprog
+
"/golang_test_defer_excchain.txt-ipython"
)
retcode
,
stdout
,
stderr
=
_pyrun
([
"-m"
,
"IPython"
,
"--quick"
,
"--colors=NoColor"
,
retcode
,
stdout
,
stderr
=
_pyrun
([
"-m"
,
"IPython"
,
"--quick"
,
"--colors=NoColor"
,
"-m"
,
"golang_test_defer_excchain"
],
"-m"
,
"golang_test_defer_excchain"
],
...
@@ -1724,8 +1739,28 @@ def _pyrun(argv, stdin=None, stdout=None, stderr=None, **kw): # -> retcode, st
...
@@ -1724,8 +1739,28 @@ def _pyrun(argv, stdin=None, stdout=None, stderr=None, **kw): # -> retcode, st
pathv
.
extend
(
envpath
.
split
(
os
.
pathsep
))
pathv
.
extend
(
envpath
.
split
(
os
.
pathsep
))
env
[
'PYTHONPATH'
]
=
os
.
pathsep
.
join
(
pathv
)
env
[
'PYTHONPATH'
]
=
os
.
pathsep
.
join
(
pathv
)
# set $PYTHONIOENCODING to encoding of stdin/stdout/stderr
# we need to do it because on Windows `python x.py | ...` runs with stdio
# encoding set to cp125X even if just `python x.py` runs with stdio
# encoding=UTF-8.
if
'PYTHONIOENCODING'
not
in
env
:
enc
=
set
([
_
.
encoding
for
_
in
(
sys
.
stdin
,
sys
.
stdout
,
sys
.
stderr
)])
if
None
in
enc
:
# without -s pytest uses _pytest.capture.DontReadFromInput
enc
.
remove
(
None
)
# with None .encoding
assert
len
(
enc
)
==
1
env
[
'PYTHONIOENCODING'
]
=
enc
.
pop
()
p
=
Popen
(
argv
,
stdin
=
(
PIPE
if
stdin
else
None
),
stdout
=
stdout
,
stderr
=
stderr
,
env
=
env
,
**
kw
)
p
=
Popen
(
argv
,
stdin
=
(
PIPE
if
stdin
else
None
),
stdout
=
stdout
,
stderr
=
stderr
,
env
=
env
,
**
kw
)
stdout
,
stderr
=
p
.
communicate
(
stdin
)
stdout
,
stderr
=
p
.
communicate
(
stdin
)
# on windows print emits \r\n instead of just \n
# normalize that to \n in *out
if
os
.
name
==
'nt'
:
if
stdout
is
not
None
:
stdout
=
stdout
.
replace
(
b'
\
r
\
n
'
,
b'
\
n
'
)
if
stderr
is
not
None
:
stderr
=
stderr
.
replace
(
b'
\
r
\
n
'
,
b'
\
n
'
)
return
p
.
returncode
,
stdout
,
stderr
return
p
.
returncode
,
stdout
,
stderr
# pyrun runs `sys.executable argv... <stdin`.
# pyrun runs `sys.executable argv... <stdin`.
...
@@ -1792,10 +1827,33 @@ def assertDoc(want, got):
...
@@ -1792,10 +1827,33 @@ def assertDoc(want, got):
got = got.replace(dir_pygolang, "
PYGOLANG
") # /home/x/.../pygolang -> PYGOLANG
got = got.replace(dir_pygolang, "
PYGOLANG
") # /home/x/.../pygolang -> PYGOLANG
got = got.replace(udir_pygolang, "
PYGOLANG
") # ~/.../pygolang -> PYGOLANG
got = got.replace(udir_pygolang, "
PYGOLANG
") # ~/.../pygolang -> PYGOLANG
# got: normalize PYGOLANG
\
a
\
b
\
c -> PYGOLANG/
a
/b/c
# a
\
b
\
c
\
d.py -> a/b/c/d.py
def _(m):
return m.group(0).replace(os.path.sep, '/')
got = re.sub(r"
(
?
<=
PYGOLANG
)[
^
\
s
]
+
(
?
=
\
s
)
", _, got)
got = re.sub(r"
([
\
w
\\\
.]
+
)(
?
=
\
.
py
)
", _, got)
# want: process conditionals
# want: process conditionals
# PY39(...) -> ... if py39 else ø
# PY39(...) -> ... if py ≥ 3.9 else ø (inline)
py39 = sys.version_info >= (3, 9)
# `... +PY39` -> ... if py ≥ 3.9 else ø (whole line)
want = re.sub(r"
PY39
\
((.
*
)
\
)
", r"
\
1
" if py39 else "", want)
# `... -PY39` -> ... if py < 3.9 else ø (whole line)
have = {} # 'PYxy' -> y/n
for minor in (9,10,11):
have['PY3%d' % minor] = (sys.version_info >= (3, minor))
for x, havex in have.items():
want = re.sub(r"
%
s
\
((.
*
)
\
)
" % x, r"
\
1
" if havex else "", want)
r = re.compile(r'^(?P<main>.*?) +(?P<y>(
\
+|-))%s$
'
% x)
v = []
for l in want.splitlines():
m = r.match(l)
if m is not None:
l = m.group('main')
y = {'+':True, '-':False}[m.group('y')]
if (y and not havex) or (havex and not y):
continue
v.append(l)
want = '
\
n
'.join(v)+'
\
n
'
# want: ^$ -> <BLANKLINE>
# want: ^$ -> <BLANKLINE>
while "
\
n
\
n
" in want:
while "
\
n
\
n
" in want:
...
@@ -1818,8 +1876,11 @@ def assertDoc(want, got):
...
@@ -1818,8 +1876,11 @@ def assertDoc(want, got):
# ...
# ...
# fmtargspec(f) -> '(x, y=3)'
# fmtargspec(f) -> '(x, y=3)'
def fmtargspec(f): # -> str
def fmtargspec(f): # -> str
with warnings.catch_warnings():
# inspect.formatargspec is deprecated since py3.5 and was removed in py3.11
warnings.simplefilter('ignore', DeprecationWarning)
# -> use inspect.signature instead.
if six.PY3:
return str(inspect.signature(f))
else:
return inspect.formatargspec(*inspect.getargspec(f))
return inspect.formatargspec(*inspect.getargspec(f))
def test_fmtargspec():
def test_fmtargspec():
...
@@ -1828,8 +1889,10 @@ def test_fmtargspec():
...
@@ -1828,8 +1889,10 @@ def test_fmtargspec():
# readfile returns content of file @path.
# readfile returns content of file @path.
def readfile(path):
def readfile(path): # -> bytes
with open(path, "r") as f:
# on windows in text mode files are opened with encoding=locale.getdefaultlocale()
# which is CP125X instead of UTF-8. -> manually decode as 'UTF-8'
with open(path, "rb") as f:
return f.read()
return f.read()
# abbrev_home returns path with user home prefix abbreviated with ~.
# abbrev_home returns path with user home prefix abbreviated with ~.
...
...
golang/libgolang.h
View file @
2b56ce71
#ifndef _NXD_LIBGOLANG_H
#ifndef _NXD_LIBGOLANG_H
#define _NXD_LIBGOLANG_H
#define _NXD_LIBGOLANG_H
// Copyright (C) 2018-202
2
Nexedi SA and Contributors.
// Copyright (C) 2018-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -176,6 +176,12 @@
...
@@ -176,6 +176,12 @@
#include <sys/types.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/stat.h>
#include <fcntl.h>
#ifdef _MSC_VER // no mode_t on msvc
typedef
int
mode_t
;
#endif
// DSO symbols visibility (based on https://gcc.gnu.org/wiki/Visibility)
// DSO symbols visibility (based on https://gcc.gnu.org/wiki/Visibility)
#if defined _WIN32 || defined __CYGWIN__
#if defined _WIN32 || defined __CYGWIN__
#define LIBGOLANG_DSO_EXPORT __declspec(dllexport)
#define LIBGOLANG_DSO_EXPORT __declspec(dllexport)
...
@@ -577,7 +583,7 @@ int select(const _selcase (&casev)[N]) {
...
@@ -577,7 +583,7 @@ int select(const _selcase (&casev)[N]) {
static
inline
// select(vector<casev>)
static
inline
// select(vector<casev>)
int
select
(
const
std
::
vector
<
_selcase
>
&
casev
)
{
int
select
(
const
std
::
vector
<
_selcase
>
&
casev
)
{
return
_chanselect
(
&
casev
[
0
]
,
casev
.
size
());
return
_chanselect
(
casev
.
data
()
,
casev
.
size
());
}
}
// defer(f) mimics `defer f()` from golang.
// defer(f) mimics `defer f()` from golang.
...
@@ -829,7 +835,7 @@ struct _interface {
...
@@ -829,7 +835,7 @@ struct _interface {
protected:
protected:
// don't use destructor -> use decref
// don't use destructor -> use decref
~
_interface
();
LIBGOLANG_API
~
_interface
();
};
};
typedef
refptr
<
_interface
>
interface
;
typedef
refptr
<
_interface
>
interface
;
...
...
golang/os.cpp
View file @
2b56ce71
// Copyright (C) 2019-202
2
Nexedi SA and Contributors.
// Copyright (C) 2019-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -37,7 +37,8 @@
...
@@ -37,7 +37,8 @@
// GLIBC < 2.32 provides sys_siglist but not sigdescr_np in its headers
// GLIBC < 2.32 provides sys_siglist but not sigdescr_np in its headers
// cut this short
// cut this short
// (on darwing sys_siglist declaration is normally provided)
// (on darwing sys_siglist declaration is normally provided)
#ifndef __APPLE__
// (on windows sys_siglist is not available at all)
#if !(defined(__APPLE__) || defined(_WIN32))
extern
"C"
{
extern
"C"
{
extern
const
char
*
const
sys_siglist
[];
extern
const
char
*
const
sys_siglist
[];
}
}
...
@@ -226,8 +227,8 @@ tuple<string, error> ReadFile(const string& path) {
...
@@ -226,8 +227,8 @@ tuple<string, error> ReadFile(const string& path) {
while
(
1
)
{
while
(
1
)
{
int
n
;
int
n
;
tie
(
n
,
err
)
=
f
->
Read
(
&
buf
[
0
]
,
buf
.
size
());
tie
(
n
,
err
)
=
f
->
Read
(
buf
.
data
()
,
buf
.
size
());
data
.
append
(
&
buf
[
0
]
,
n
);
data
.
append
(
buf
.
data
()
,
n
);
if
(
err
!=
nil
)
{
if
(
err
!=
nil
)
{
if
(
err
==
io
::
EOF_
)
if
(
err
==
io
::
EOF_
)
err
=
nil
;
err
=
nil
;
...
@@ -248,7 +249,7 @@ tuple<string, error> ReadFile(const string& path) {
...
@@ -248,7 +249,7 @@ tuple<string, error> ReadFile(const string& path) {
tuple
<
File
,
File
,
error
>
Pipe
()
{
tuple
<
File
,
File
,
error
>
Pipe
()
{
int
vfd
[
2
],
syserr
;
int
vfd
[
2
],
syserr
;
syserr
=
sys
::
Pipe
(
vfd
);
syserr
=
sys
::
Pipe
(
vfd
,
0
);
if
(
syserr
!=
0
)
if
(
syserr
!=
0
)
return
make_tuple
(
nil
,
nil
,
fmt
::
errorf
(
"pipe: %w"
,
sys
::
NewErrno
(
syserr
)));
return
make_tuple
(
nil
,
nil
,
fmt
::
errorf
(
"pipe: %w"
,
sys
::
NewErrno
(
syserr
)));
...
@@ -286,8 +287,20 @@ string Signal::String() const {
...
@@ -286,8 +287,20 @@ string Signal::String() const {
const
Signal
&
sig
=
*
this
;
const
Signal
&
sig
=
*
this
;
const
char
*
sigstr
=
nil
;
const
char
*
sigstr
=
nil
;
#ifdef _WIN32
switch
(
sig
.
signo
)
{
case
SIGABRT
:
return
"Aborted"
;
case
SIGBREAK
:
return
"Break"
;
case
SIGFPE
:
return
"Floating point exception"
;
case
SIGILL
:
return
"Illegal instruction"
;
case
SIGINT
:
return
"Interrupt"
;
case
SIGSEGV
:
return
"Segmentation fault"
;
case
SIGTERM
:
return
"Terminated"
;
}
#else
if
(
0
<=
sig
.
signo
&&
sig
.
signo
<
NSIG
)
if
(
0
<=
sig
.
signo
&&
sig
.
signo
<
NSIG
)
sigstr
=
::
sys_siglist
[
sig
.
signo
];
// might be nil as well
sigstr
=
::
sys_siglist
[
sig
.
signo
];
// might be nil as well
#endif
if
(
sigstr
!=
nil
)
if
(
sigstr
!=
nil
)
return
string
(
sigstr
);
return
string
(
sigstr
);
...
...
golang/os.h
View file @
2b56ce71
#ifndef _NXD_LIBGOLANG_OS_H
#ifndef _NXD_LIBGOLANG_OS_H
#define _NXD_LIBGOLANG_OS_H
#define _NXD_LIBGOLANG_OS_H
//
//
// Copyright (C) 2019-202
2
Nexedi SA and Contributors.
// Copyright (C) 2019-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -61,7 +61,7 @@ private:
...
@@ -61,7 +61,7 @@ private:
~
_File
();
~
_File
();
friend
File
_newFile
(
_libgolang_ioh
*
ioh
,
const
string
&
name
);
friend
File
_newFile
(
_libgolang_ioh
*
ioh
,
const
string
&
name
);
public:
public:
void
decref
();
LIBGOLANG_API
void
decref
();
public:
public:
LIBGOLANG_API
string
Name
()
const
;
LIBGOLANG_API
string
Name
()
const
;
...
@@ -95,9 +95,15 @@ private:
...
@@ -95,9 +95,15 @@ private:
// Open opens file @path.
// Open opens file @path.
LIBGOLANG_API
std
::
tuple
<
File
,
error
>
Open
(
const
string
&
path
,
int
flags
=
O_RDONLY
,
LIBGOLANG_API
std
::
tuple
<
File
,
error
>
Open
(
const
string
&
path
,
int
flags
=
O_RDONLY
,
mode_t
mode
=
S_IRUSR
|
S_IWUSR
|
S_IXUSR
|
mode_t
mode
=
#if !defined(_MSC_VER)
S_IRUSR
|
S_IWUSR
|
S_IXUSR
|
S_IRGRP
|
S_IWGRP
|
S_IXGRP
|
S_IRGRP
|
S_IWGRP
|
S_IXGRP
|
S_IROTH
|
S_IWOTH
|
S_IXOTH
);
S_IROTH
|
S_IWOTH
|
S_IXOTH
#else
_S_IREAD
|
_S_IWRITE
#endif
);
// NewFile wraps OS-level file-descriptor into File.
// NewFile wraps OS-level file-descriptor into File.
// The ownership of sysfd is transferred to File.
// The ownership of sysfd is transferred to File.
...
...
golang/os/signal.cpp
View file @
2b56ce71
// Copyright (C) 2021-202
2
Nexedi SA and Contributors.
// Copyright (C) 2021-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -89,6 +89,10 @@
...
@@ -89,6 +89,10 @@
#include <atomic>
#include <atomic>
#include <tuple>
#include <tuple>
#if defined(_WIN32)
# include <windows.h>
#endif
#define DEBUG 0
#define DEBUG 0
#if DEBUG
#if DEBUG
...
@@ -97,6 +101,12 @@
...
@@ -97,6 +101,12 @@
# define debugf(format, ...) do {} while (0)
# define debugf(format, ...) do {} while (0)
#endif
#endif
#if defined(_MSC_VER)
# define HAVE_SIGACTION 0
#else
# define HAVE_SIGACTION 1
#endif
// golang::os::signal::
// golang::os::signal::
namespace
golang
{
namespace
golang
{
namespace
os
{
namespace
os
{
...
@@ -109,13 +119,30 @@ using std::tie;
...
@@ -109,13 +119,30 @@ using std::tie;
using
std
::
vector
;
using
std
::
vector
;
using
cxx
::
set
;
using
cxx
::
set
;
#if HAVE_SIGACTION
static
void
xsigemptyset
(
sigset_t
*
sa_mask
);
#else
// custom `struct sigaction` emulated via signal(2)
#define SA_SIGINFO 1
typedef
struct
{}
siginfo_t
;
struct
sigaction
{
union
{
void
(
*
sa_handler
)(
int
);
void
(
*
sa_sigaction
)(
int
,
siginfo_t
*
,
void
*
);
};
int
sa_flags
;
};
static
void
_os_sighandler_nosigaction
(
int
sig
);
#endif
static
void
_os_sighandler
(
int
sig
,
siginfo_t
*
info
,
void
*
ucontext
);
static
void
_os_sighandler
(
int
sig
,
siginfo_t
*
info
,
void
*
ucontext
);
static
void
_notify
(
int
signo
);
static
void
_notify
(
int
signo
);
static
void
_checksig
(
int
signo
);
static
void
_checksig
(
int
signo
);
static
void
_checkActEqual
(
const
struct
sigaction
*
a
,
const
struct
sigaction
*
b
);
static
void
_checkActEqual
(
const
struct
sigaction
*
a
,
const
struct
sigaction
*
b
);
static
void
_spinwaitNextQueueCycle
();
static
void
_spinwaitNextQueueCycle
();
static
int
sys_sigaction
(
int
signo
,
const
struct
sigaction
*
act
,
struct
sigaction
*
oldact
);
static
void
xsys_sigaction
(
int
signo
,
const
struct
sigaction
*
act
,
struct
sigaction
*
oldact
);
static
void
xsys_sigaction
(
int
signo
,
const
struct
sigaction
*
act
,
struct
sigaction
*
oldact
);
static
void
xsigemptyset
(
sigset_t
*
sa_mask
);
static
bool
_sigact_equal
(
const
struct
sigaction
*
a
,
const
struct
sigaction
*
b
);
static
bool
_sigact_equal
(
const
struct
sigaction
*
a
,
const
struct
sigaction
*
b
);
...
@@ -160,27 +187,31 @@ void _init() {
...
@@ -160,27 +187,31 @@ void _init() {
// create _wakerx <-> _waketx pipe; set _waketx to nonblocking mode
// create _wakerx <-> _waketx pipe; set _waketx to nonblocking mode
int
vfd
[
2
];
int
vfd
[
2
];
if
(
sys
::
Pipe
(
vfd
)
<
0
)
if
(
sys
::
Pipe
(
vfd
,
O_CLOEXEC
)
<
0
)
panic
(
"pipe(_wakerx, _waketx)"
);
// TODO +syserr
panic
(
"pipe(_wakerx, _waketx)"
);
// TODO +syserr
if
(
sys
::
Fcntl
(
vfd
[
0
],
F_SETFD
,
FD_CLOEXEC
)
<
0
)
panic
(
"fcntl(_wakerx, FD_CLOEXEC)"
);
// TODO +syserr
error
err
;
error
err
;
tie
(
_wakerx
,
err
)
=
os
::
NewFile
(
vfd
[
0
],
"_wakerx"
);
tie
(
_wakerx
,
err
)
=
os
::
NewFile
(
vfd
[
0
],
"_wakerx"
);
if
(
err
!=
nil
)
if
(
err
!=
nil
)
panic
(
"os::newFile(_wakerx"
);
panic
(
"os::newFile(_wakerx"
);
_waketx
=
vfd
[
1
];
_waketx
=
vfd
[
1
];
#ifndef _WIN32
if
(
sys
::
Fcntl
(
_waketx
,
F_SETFL
,
O_NONBLOCK
)
<
0
)
if
(
sys
::
Fcntl
(
_waketx
,
F_SETFL
,
O_NONBLOCK
)
<
0
)
panic
(
"fcntl(_waketx, O_NONBLOCK)"
);
// TODO +syserr
panic
(
"fcntl(_waketx, O_NONBLOCK)"
);
// TODO +syserr
if
(
sys
::
Fcntl
(
_waketx
,
F_SETFD
,
FD_CLOEXEC
)
<
0
)
#else
panic
(
"fcntl(_waketx, FD_CLOEXEC)"
);
// TODO +syserr
HANDLE
hwaketx
=
(
HANDLE
)
_get_osfhandle
(
_waketx
);
DWORD
mode
=
PIPE_READMODE_BYTE
|
PIPE_NOWAIT
;
if
(
!
SetNamedPipeHandleState
(
hwaketx
,
&
mode
,
NULL
,
NULL
))
panic
(
"SetNamedPipeHandleState(hwaketx, PIPE_NOWAIT)"
);
// TODO +syserr
#endif
_actIgnore
.
sa_handler
=
SIG_IGN
;
_actIgnore
.
sa_handler
=
SIG_IGN
;
_actIgnore
.
sa_flags
=
0
;
_actIgnore
.
sa_flags
=
0
;
xsigemptyset
(
&
_actIgnore
.
sa_mask
);
_actNotify
.
sa_sigaction
=
_os_sighandler
;
_actNotify
.
sa_sigaction
=
_os_sighandler
;
_actNotify
.
sa_flags
=
SA_SIGINFO
;
_actNotify
.
sa_flags
=
SA_SIGINFO
;
#if HAVE_SIGACTION
xsigemptyset
(
&
_actIgnore
.
sa_mask
);
xsigemptyset
(
&
_actNotify
.
sa_mask
);
xsigemptyset
(
&
_actNotify
.
sa_mask
);
#endif
}
}
...
@@ -248,6 +279,14 @@ done:
...
@@ -248,6 +279,14 @@ done:
if
(
sah
!=
SIG_IGN
)
{
if
(
sah
!=
SIG_IGN
)
{
if
(
sah
!=
SIG_DFL
)
{
if
(
sah
!=
SIG_DFL
)
{
sah
(
sig
);
sah
(
sig
);
// handlers installed via signal usually reinstall themselves
// return it back to us
struct
sigaction
after
;
xsys_sigaction
(
sig
,
&
_actNotify
,
&
after
);
if
(
!
(
_sigact_equal
(
&
after
,
&
_actNotify
)
||
(
((
after
.
sa_flags
&
SA_SIGINFO
)
==
0
)
&&
((
after
.
sa_handler
==
SIG_DFL
)
||
(
after
.
sa_handler
==
sah
)))))
panic
(
"collision detected wrt thirdparty signal usage"
);
}
}
else
{
else
{
// SIG_DFL && _SigReset - reraise to die if the signal is fatal
// SIG_DFL && _SigReset - reraise to die if the signal is fatal
...
@@ -343,7 +382,7 @@ static int/*syserr*/ _Notify1(chan<os::Signal> ch, os::Signal sig) {
...
@@ -343,7 +382,7 @@ static int/*syserr*/ _Notify1(chan<os::Signal> ch, os::Signal sig) {
// retrieve current signal action
// retrieve current signal action
struct
sigaction
cur
;
struct
sigaction
cur
;
int
syserr
=
sys
::
S
igaction
(
sig
.
signo
,
nil
,
&
cur
);
int
syserr
=
sys
_s
igaction
(
sig
.
signo
,
nil
,
&
cur
);
if
(
syserr
<
0
)
{
if
(
syserr
<
0
)
{
// TODO reenable once we can panic with any object
// TODO reenable once we can panic with any object
//return fmt::errorf("sigaction sig%d: %w", sig.signo, sys::NewErrno(syserr);
//return fmt::errorf("sigaction sig%d: %w", sig.signo, sys::NewErrno(syserr);
...
@@ -389,7 +428,7 @@ static int/*syserr*/ _Notify1(chan<os::Signal> ch, os::Signal sig) {
...
@@ -389,7 +428,7 @@ static int/*syserr*/ _Notify1(chan<os::Signal> ch, os::Signal sig) {
// register our sigaction
// register our sigaction
struct
sigaction
old
;
struct
sigaction
old
;
syserr
=
sys
::
S
igaction
(
sig
.
signo
,
&
_actNotify
,
&
old
);
syserr
=
sys
_s
igaction
(
sig
.
signo
,
&
_actNotify
,
&
old
);
if
(
syserr
<
0
)
{
if
(
syserr
<
0
)
{
// TODO reenable once we can panic with any object
// TODO reenable once we can panic with any object
//return fmt::errorf("sigaction sig%d: %w", sig.signo, sys::NewErrno(syserr);
//return fmt::errorf("sigaction sig%d: %w", sig.signo, sys::NewErrno(syserr);
...
@@ -469,7 +508,7 @@ static int/*syserr*/ _Ignore1(os::Signal sig) {
...
@@ -469,7 +508,7 @@ static int/*syserr*/ _Ignore1(os::Signal sig) {
if
(
h
==
nil
)
{
if
(
h
==
nil
)
{
h
=
new
_SigHandler
();
h
=
new
_SigHandler
();
h
->
sigstate
.
store
(
_SigIgnoring
);
h
->
sigstate
.
store
(
_SigIgnoring
);
int
syserr
=
sys
::
S
igaction
(
sig
.
signo
,
nil
,
&
h
->
prev_act
);
int
syserr
=
sys
_s
igaction
(
sig
.
signo
,
nil
,
&
h
->
prev_act
);
if
(
syserr
<
0
)
{
if
(
syserr
<
0
)
{
delete
h
;
delete
h
;
return
syserr
;
// TODO errctx
return
syserr
;
// TODO errctx
...
@@ -481,7 +520,7 @@ static int/*syserr*/ _Ignore1(os::Signal sig) {
...
@@ -481,7 +520,7 @@ static int/*syserr*/ _Ignore1(os::Signal sig) {
h
->
sigstate
.
store
(
_SigIgnoring
);
h
->
sigstate
.
store
(
_SigIgnoring
);
h
->
subscribers
.
clear
();
h
->
subscribers
.
clear
();
int
syserr
=
sys
::
S
igaction
(
sig
.
signo
,
&
_actIgnore
,
nil
);
int
syserr
=
sys
_s
igaction
(
sig
.
signo
,
&
_actIgnore
,
nil
);
if
(
syserr
<
0
)
if
(
syserr
<
0
)
return
syserr
;
// TODO errctx
return
syserr
;
// TODO errctx
...
@@ -509,7 +548,7 @@ static int/*syserr*/ _Reset1(os::Signal sig) {
...
@@ -509,7 +548,7 @@ static int/*syserr*/ _Reset1(os::Signal sig) {
h
->
sigstate
.
store
(
_SigReset
);
h
->
sigstate
.
store
(
_SigReset
);
struct
sigaction
act
;
struct
sigaction
act
;
int
syserr
=
sys
::
S
igaction
(
sig
.
signo
,
&
h
->
prev_act
,
&
act
);
int
syserr
=
sys
_s
igaction
(
sig
.
signo
,
&
h
->
prev_act
,
&
act
);
if
(
syserr
<
0
)
if
(
syserr
<
0
)
return
syserr
;
// TODO errctx
return
syserr
;
// TODO errctx
if
(
sigstate
==
_SigNotifying
)
if
(
sigstate
==
_SigNotifying
)
...
@@ -617,16 +656,75 @@ static void _checksig(int signo) {
...
@@ -617,16 +656,75 @@ static void _checksig(int signo) {
}
}
#if HAVE_SIGACTION
static
void
xsigemptyset
(
sigset_t
*
sa_mask
)
{
static
void
xsigemptyset
(
sigset_t
*
sa_mask
)
{
if
(
sigemptyset
(
sa_mask
)
<
0
)
if
(
sigemptyset
(
sa_mask
)
<
0
)
panic
(
"sigemptyset failed"
);
// must always succeed
panic
(
"sigemptyset failed"
);
// must always succeed
}
}
#endif
static
void
xsys_sigaction
(
int
signo
,
const
struct
sigaction
*
act
,
struct
sigaction
*
oldact
)
{
static
void
xsys_sigaction
(
int
signo
,
const
struct
sigaction
*
act
,
struct
sigaction
*
oldact
)
{
int
syserr
=
sys
::
S
igaction
(
signo
,
act
,
oldact
);
int
syserr
=
sys
_s
igaction
(
signo
,
act
,
oldact
);
if
(
syserr
!=
0
)
if
(
syserr
!=
0
)
panic
(
"sigaction failed"
);
// TODO add errno detail
panic
(
"sigaction failed"
);
// TODO add errno detail
}
}
static
int
sys_sigaction
(
int
signo
,
const
struct
sigaction
*
act
,
struct
sigaction
*
oldact
)
{
#if HAVE_SIGACTION
return
sys
::
Sigaction
(
signo
,
act
,
oldact
);
#else
sys
::
sighandler_t
oldh
,
newh
;
int
syserr
;
if
(
act
==
nil
)
{
newh
=
SIG_GET
;
}
else
if
(
act
->
sa_flags
&
SA_SIGINFO
)
{
if
(
act
->
sa_sigaction
!=
_os_sighandler
)
panic
(
"BUG: compat sigaction: act->sa_sigaction != _os_sighandler"
);
newh
=
_os_sighandler_nosigaction
;
}
else
{
newh
=
act
->
sa_handler
;
}
oldh
=
sys
::
Signal
(
signo
,
newh
,
&
syserr
);
if
(
oldh
==
SIG_ERR
)
return
syserr
;
if
(
oldact
!=
nil
)
{
if
(
oldh
==
_os_sighandler_nosigaction
)
{
oldact
->
sa_sigaction
=
_os_sighandler
;
oldact
->
sa_flags
=
SA_SIGINFO
;
}
else
{
oldact
->
sa_handler
=
oldh
;
oldact
->
sa_flags
=
0
;
}
}
return
0
;
#endif
}
#if !HAVE_SIGACTION
static
void
_os_sighandler_nosigaction
(
int
sig
)
{
// reinstall sighandler since on handler invocation it is reset to SIG_DFL.
// do it as fast as we can as the first thing - to minimize window of race
// while signal handler is reset from being our _os_sighandler.
//
// NOTE it is ok to reinstall before invoking _os_sighandler because
// _os_sighandler uses only atomics and pipe write and so is reentrant.
sys
::
sighandler_t
h
;
int
syserr
;
h
=
sys
::
Signal
(
sig
,
_os_sighandler_nosigaction
,
&
syserr
);
if
(
h
==
SIG_ERR
)
panic
(
"signal(reinstall) failed"
);
if
(
!
(
h
==
SIG_DFL
/*reset*/
||
h
==
_os_sighandler_nosigaction
/*concurrent sighandler*/
)
)
panic
(
"collision detected wrt thirdparty signal usage"
);
// invoke _os_sighandler as if it was setup by sigaction.
_os_sighandler
(
sig
,
NULL
,
NULL
);
}
#endif
static
bool
_sigact_equal
(
const
struct
sigaction
*
a
,
const
struct
sigaction
*
b
)
{
static
bool
_sigact_equal
(
const
struct
sigaction
*
a
,
const
struct
sigaction
*
b
)
{
// don't compare sigaction by memcmp - it will fail because struct sigaction
// don't compare sigaction by memcmp - it will fail because struct sigaction
...
...
golang/os/signal_test.py
View file @
2b56ce71
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
# Copyright (C) 2021-202
2
Nexedi SA and Contributors.
# Copyright (C) 2021-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -30,12 +30,27 @@ from golang.golang_test import panics, _pyrun
...
@@ -30,12 +30,27 @@ from golang.golang_test import panics, _pyrun
from
pytest
import
raises
from
pytest
import
raises
from
subprocess
import
PIPE
from
subprocess
import
PIPE
try
:
from
signal
import
raise_signal
except
ImportError
:
# py2
from
_testcapi
import
raise_signal
# directories
# directories
dir_os
=
dirname
(
__file__
)
# .../pygolang/os
dir_os
=
dirname
(
__file__
)
# .../pygolang/os
dir_testprog
=
dir_os
+
"/testprog"
# .../pygolang/os/testprog
dir_testprog
=
dir_os
+
"/testprog"
# .../pygolang/os/testprog
# default to use SIGUSR1/SIGUSR2 in tests.
# but use SIGTERM/SIGINT if those are not available (windows).
try
:
SIG1
=
getattr
(
syscall
,
'SIGUSR1'
)
SIG2
=
getattr
(
syscall
,
'SIGUSR2'
)
except
AttributeError
:
SIG1
=
syscall
.
SIGTERM
SIG2
=
syscall
.
SIGINT
N
=
1000
N
=
1000
# test_signal verifies signal delivery to channels controlled by Notify/Stop/Ignore/Reset.
# test_signal verifies signal delivery to channels controlled by Notify/Stop/Ignore/Reset.
...
@@ -43,9 +58,9 @@ N = 1000
...
@@ -43,9 +58,9 @@ N = 1000
def
test_signal
():
def
test_signal
():
# Notify/Stop with wrong chan dtype -> panic
# Notify/Stop with wrong chan dtype -> panic
_
=
panics
(
"pychan: channel type mismatch"
)
_
=
panics
(
"pychan: channel type mismatch"
)
with
_
:
signal
.
Notify
(
chan
(
2
),
syscall
.
SIGUSR
1
)
with
_
:
signal
.
Notify
(
chan
(
2
),
SIG
1
)
with
_
:
signal
.
Stop
(
chan
(
2
))
with
_
:
signal
.
Stop
(
chan
(
2
))
with
_
:
signal
.
Notify
(
chan
(
2
,
dtype
=
'C.int'
),
syscall
.
SIGUSR
1
)
with
_
:
signal
.
Notify
(
chan
(
2
,
dtype
=
'C.int'
),
SIG
1
)
with
_
:
signal
.
Stop
(
chan
(
2
,
dtype
=
'C.int'
))
with
_
:
signal
.
Stop
(
chan
(
2
,
dtype
=
'C.int'
))
# Notify/Ignore/Reset with wrong signal type
# Notify/Ignore/Reset with wrong signal type
...
@@ -54,109 +69,109 @@ def test_signal():
...
@@ -54,109 +69,109 @@ def test_signal():
with
_
:
signal
.
Ignore
(
None
)
with
_
:
signal
.
Ignore
(
None
)
with
_
:
signal
.
Reset
(
None
)
with
_
:
signal
.
Reset
(
None
)
# subscribe ch1(
USR1), ch12(USR1,USR2) and ch2(USR
2)
# subscribe ch1(
SIG1), ch12(SIG1,SIG2) and ch2(SIG
2)
ch1
=
chan
(
2
,
dtype
=
gos
.
Signal
)
ch1
=
chan
(
2
,
dtype
=
gos
.
Signal
)
ch12
=
chan
(
2
,
dtype
=
gos
.
Signal
)
ch12
=
chan
(
2
,
dtype
=
gos
.
Signal
)
ch2
=
chan
(
2
,
dtype
=
gos
.
Signal
)
ch2
=
chan
(
2
,
dtype
=
gos
.
Signal
)
signal
.
Notify
(
ch1
,
syscall
.
SIGUSR
1
)
signal
.
Notify
(
ch1
,
SIG
1
)
signal
.
Notify
(
ch12
,
syscall
.
SIGUSR1
,
syscall
.
SIGUSR
2
)
signal
.
Notify
(
ch12
,
SIG1
,
SIG
2
)
signal
.
Notify
(
ch2
,
syscall
.
SIGUSR
2
)
signal
.
Notify
(
ch2
,
SIG
2
)
def
_
():
def
_
():
signal
.
Reset
()
signal
.
Reset
()
defer
(
_
)
defer
(
_
)
for
i
in
range
(
N
):
for
i
in
range
(
N
):
# raise SIG
USR
1 -> should be delivered to ch1 and ch12
# raise SIG1 -> should be delivered to ch1 and ch12
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
and
len
(
ch12
)
==
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
and
len
(
ch12
)
==
1
)
sig1
=
ch1
.
recv
()
sig1
=
ch1
.
recv
()
sig12
=
ch12
.
recv
()
sig12
=
ch12
.
recv
()
assert
sig1
==
syscall
.
SIGUSR
1
assert
sig1
==
SIG
1
assert
sig12
==
syscall
.
SIGUSR
1
assert
sig12
==
SIG
1
# raise SIG
USR
2 -> should be delivered to ch12 and ch2
# raise SIG2 -> should be delivered to ch12 and ch2
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
2
)
killme
(
SIG
2
)
waitfor
(
lambda
:
len
(
ch12
)
==
1
and
len
(
ch2
)
==
1
)
waitfor
(
lambda
:
len
(
ch12
)
==
1
and
len
(
ch2
)
==
1
)
sig12
=
ch12
.
recv
()
sig12
=
ch12
.
recv
()
sig2
=
ch2
.
recv
()
sig2
=
ch2
.
recv
()
assert
sig12
==
syscall
.
SIGUSR
2
assert
sig12
==
SIG
2
assert
sig2
==
syscall
.
SIGUSR
2
assert
sig2
==
SIG
2
# if SIG
USR
2 will be eventually delivered to ch1 - it will break
# if SIG2 will be eventually delivered to ch1 - it will break
# in SIG
USR
1 check on next iteration.
# in SIG1 check on next iteration.
# Stop(ch2) -> signals should not be delivered to ch2 anymore
# Stop(ch2) -> signals should not be delivered to ch2 anymore
signal
.
Stop
(
ch2
)
signal
.
Stop
(
ch2
)
for
i
in
range
(
N
):
for
i
in
range
(
N
):
#
USR
1 -> ch1, ch12
#
SIG
1 -> ch1, ch12
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
and
len
(
ch12
)
==
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
and
len
(
ch12
)
==
1
)
sig1
=
ch1
.
recv
()
sig1
=
ch1
.
recv
()
sig12
=
ch12
.
recv
()
sig12
=
ch12
.
recv
()
assert
sig1
==
syscall
.
SIGUSR
1
assert
sig1
==
SIG
1
assert
sig12
==
syscall
.
SIGUSR
1
assert
sig12
==
SIG
1
#
USR
2 -> ch12, !ch2
#
SIG
2 -> ch12, !ch2
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
2
)
killme
(
SIG
2
)
waitfor
(
lambda
:
len
(
ch12
)
==
1
)
waitfor
(
lambda
:
len
(
ch12
)
==
1
)
sig12
=
ch12
.
recv
()
sig12
=
ch12
.
recv
()
assert
sig12
==
syscall
.
SIGUSR
2
assert
sig12
==
SIG
2
# if SIG
USR
2 will be eventually delivered to ch2 - it will break on
# if SIG2 will be eventually delivered to ch2 - it will break on
# next iteration.
# next iteration.
# Ignore(
USR1) -> ch1 should not be delivered to anymore, ch12 should be delivered only USR
2
# Ignore(
SIG1) -> ch1 should not be delivered to anymore, ch12 should be delivered only SIG
2
signal
.
Ignore
(
syscall
.
SIGUSR
1
)
signal
.
Ignore
(
SIG
1
)
for
i
in
range
(
N
):
for
i
in
range
(
N
):
#
USR
1 -> ø
#
SIG
1 -> ø
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
time
.
sleep
(
1E-6
)
time
.
sleep
(
1E-6
)
#
USR
2 -> ch12
#
SIG
2 -> ch12
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
2
)
killme
(
SIG
2
)
waitfor
(
lambda
:
len
(
ch12
)
==
1
)
waitfor
(
lambda
:
len
(
ch12
)
==
1
)
sig12
=
ch12
.
recv
()
sig12
=
ch12
.
recv
()
assert
sig12
==
syscall
.
SIGUSR
2
assert
sig12
==
SIG
2
# if SIG
USR1 or SIGUSR
2 will be eventually delivered to ch1 or ch2 - it
# if SIG
1 or SIG
2 will be eventually delivered to ch1 or ch2 - it
# will break on next iteration.
# will break on next iteration.
# Notify after Ignore
# Notify after Ignore
signal
.
Notify
(
ch1
,
syscall
.
SIGUSR
1
)
signal
.
Notify
(
ch1
,
SIG
1
)
for
i
in
range
(
N
):
for
i
in
range
(
N
):
#
USR
1 -> ch1
#
SIG
1 -> ch1
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
)
sig1
=
ch1
.
recv
()
sig1
=
ch1
.
recv
()
assert
sig1
==
syscall
.
SIGUSR
1
assert
sig1
==
SIG
1
#
USR
2 -> ch12
#
SIG
2 -> ch12
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch12
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
2
)
killme
(
SIG
2
)
waitfor
(
lambda
:
len
(
ch12
)
==
1
)
waitfor
(
lambda
:
len
(
ch12
)
==
1
)
sig12
=
ch12
.
recv
()
sig12
=
ch12
.
recv
()
assert
sig12
==
syscall
.
SIGUSR
2
assert
sig12
==
SIG
2
# if SIG
USR1 or SIGUSR
2 will be eventually delivered to wrong place -
# if SIG
1 or SIG
2 will be eventually delivered to wrong place -
# it will break on next iteration.
# it will break on next iteration.
# Reset is tested in test_stdlib_interop (it needs non-terminating default
# Reset is tested in test_stdlib_interop (it needs non-terminating default
...
@@ -173,11 +188,11 @@ def test_notify_reinstall():
...
@@ -173,11 +188,11 @@ def test_notify_reinstall():
for
i
in
range
(
N
):
for
i
in
range
(
N
):
signal
.
Stop
(
ch
)
signal
.
Stop
(
ch
)
signal
.
Notify
(
ch
,
syscall
.
SIGUSR
1
)
signal
.
Notify
(
ch
,
SIG
1
)
time
.
sleep
(
0.1
*
time
.
second
)
time
.
sleep
(
0.1
*
time
.
second
)
assert
len
(
ch
)
==
0
assert
len
(
ch
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
time
.
sleep
(
0.1
*
time
.
second
)
time
.
sleep
(
0.1
*
time
.
second
)
assert
len
(
ch
)
==
1
assert
len
(
ch
)
==
1
...
@@ -188,7 +203,7 @@ def test_signal_all():
...
@@ -188,7 +203,7 @@ def test_signal_all():
assert
b"ok (notify)"
in
out
assert
b"ok (notify)"
in
out
assert
b"ok (ignore)"
in
out
assert
b"ok (ignore)"
in
out
assert
b"terminating ..."
in
out
assert
b"terminating ..."
in
out
assert
retcode
==
-
syscall
.
SIGTERM
.
signo
assert
retcode
==
(
-
syscall
.
SIGTERM
.
signo
if
os
.
name
!=
'nt'
else
3
)
# test_stdlib_interop verifies that there is minimal compatibility in between
# test_stdlib_interop verifies that there is minimal compatibility in between
...
@@ -206,23 +221,23 @@ def test_stdlib_interop():
...
@@ -206,23 +221,23 @@ def test_stdlib_interop():
ch1
=
chan
(
2
,
dtype
=
object
)
# NOTE not gos.Signal nor 'C.os::Signal'
ch1
=
chan
(
2
,
dtype
=
object
)
# NOTE not gos.Signal nor 'C.os::Signal'
def
_
(
signo
,
frame
):
def
_
(
signo
,
frame
):
ch1
.
send
(
"
USR
1"
)
ch1
.
send
(
"
SIG
1"
)
pysig
.
signal
(
pysig
.
SIGUSR1
,
_
)
pysig
.
signal
(
SIG1
.
signo
,
_
)
def
_
():
def
_
():
pysig
.
signal
(
pysig
.
SIGUSR1
,
pysig
.
SIG_IGN
)
pysig
.
signal
(
SIG1
.
signo
,
pysig
.
SIG_IGN
)
defer
(
_
)
defer
(
_
)
# verify that plain pysig delivery works
# verify that plain pysig delivery works
for
i
in
range
(
N
):
for
i
in
range
(
N
):
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
)
obj1
=
ch1
.
recv
()
obj1
=
ch1
.
recv
()
assert
obj1
==
"
USR
1"
assert
obj1
==
"
SIG
1"
# verify that combined pysig + golang.os.signal delivery works
# verify that combined pysig + golang.os.signal delivery works
ch2
=
chan
(
2
,
dtype
=
gos
.
Signal
)
ch2
=
chan
(
2
,
dtype
=
gos
.
Signal
)
signal
.
Notify
(
ch2
,
syscall
.
SIGUSR
1
)
signal
.
Notify
(
ch2
,
SIG
1
)
def
_
():
def
_
():
signal
.
Stop
(
ch2
)
signal
.
Stop
(
ch2
)
defer
(
_
)
defer
(
_
)
...
@@ -230,46 +245,46 @@ def test_stdlib_interop():
...
@@ -230,46 +245,46 @@ def test_stdlib_interop():
for
i
in
range
(
N
):
for
i
in
range
(
N
):
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
and
len
(
ch2
)
==
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
and
len
(
ch2
)
==
1
)
obj1
=
ch1
.
recv
()
obj1
=
ch1
.
recv
()
sig2
=
ch2
.
recv
()
sig2
=
ch2
.
recv
()
assert
obj1
==
"
USR
1"
assert
obj1
==
"
SIG
1"
assert
sig2
==
syscall
.
SIGUSR
1
assert
sig2
==
SIG
1
# Ignore stops delivery to both pysig and golang.os.signal
# Ignore stops delivery to both pysig and golang.os.signal
signal
.
Ignore
(
syscall
.
SIGUSR
1
)
signal
.
Ignore
(
SIG
1
)
for
i
in
range
(
N
):
for
i
in
range
(
N
):
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
time
.
sleep
(
1E-6
)
time
.
sleep
(
1E-6
)
time
.
sleep
(
0.1
)
# just in case
time
.
sleep
(
0.1
)
# just in case
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
# after Reset pysig delivery is restored even after Ignore
# after Reset pysig delivery is restored even after Ignore
signal
.
Reset
(
syscall
.
SIGUSR
1
)
signal
.
Reset
(
SIG
1
)
for
i
in
range
(
N
):
for
i
in
range
(
N
):
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
)
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
obj1
=
ch1
.
recv
()
obj1
=
ch1
.
recv
()
assert
obj1
==
"
USR
1"
assert
obj1
==
"
SIG
1"
# Reset stops delivery to golang.os.signal and restores pysig delivery
# Reset stops delivery to golang.os.signal and restores pysig delivery
signal
.
Notify
(
ch2
,
syscall
.
SIGUSR
1
)
signal
.
Notify
(
ch2
,
SIG
1
)
signal
.
Reset
(
syscall
.
SIGUSR
1
)
signal
.
Reset
(
SIG
1
)
for
i
in
range
(
N
):
for
i
in
range
(
N
):
assert
len
(
ch1
)
==
0
assert
len
(
ch1
)
==
0
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
killme
(
syscall
.
SIGUSR
1
)
killme
(
SIG
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
)
waitfor
(
lambda
:
len
(
ch1
)
==
1
)
assert
len
(
ch2
)
==
0
assert
len
(
ch2
)
==
0
obj1
=
ch1
.
recv
()
obj1
=
ch1
.
recv
()
assert
obj1
==
"
USR
1"
assert
obj1
==
"
SIG
1"
# test_stdlib_interop_KeyboardInterrupt verifies that signal.{Notify,Ignore} disable
# test_stdlib_interop_KeyboardInterrupt verifies that signal.{Notify,Ignore} disable
...
@@ -353,8 +368,9 @@ def test_stdlib_interop_KeyboardInterrupt():
...
@@ -353,8 +368,9 @@ def test_stdlib_interop_KeyboardInterrupt():
# killme sends signal sig to own process.
# killme sends signal sig to own process.
def
killme
(
sig
):
def
killme
(
sig
):
mypid
=
os
.
getpid
()
# use raise(sig) instead of kill(mypid, sig) so that it works on windows,
os
.
kill
(
mypid
,
sig
.
signo
)
# where os.kill unconditionally terminates target process.
raise_signal
(
sig
.
signo
)
# wait for waits until cond() becomes true or timeout.
# wait for waits until cond() becomes true or timeout.
def
waitfor
(
cond
):
def
waitfor
(
cond
):
...
...
golang/os/testprog/signal_test_all.py
View file @
2b56ce71
#!/usr/bin/env python
#!/usr/bin/env python
# Copyright (C) 2021-202
2
Nexedi SA and Contributors.
# Copyright (C) 2021-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -24,7 +24,8 @@ from __future__ import print_function, absolute_import
...
@@ -24,7 +24,8 @@ from __future__ import print_function, absolute_import
from
golang
import
chan
from
golang
import
chan
from
golang
import
os
as
gos
,
syscall
,
time
from
golang
import
os
as
gos
,
syscall
,
time
from
golang.os
import
signal
from
golang.os
import
signal
import
os
,
sys
from
golang.os.signal_test
import
killme
import
sys
def
main
():
def
main
():
# build "all signals" list
# build "all signals" list
...
@@ -35,11 +36,17 @@ def main():
...
@@ -35,11 +36,17 @@ def main():
if
sig
not
in
allsigv
:
# avoid e.g. SIGCHLD/SIGCLD dups
if
sig
not
in
allsigv
:
# avoid e.g. SIGCHLD/SIGCLD dups
allsigv
.
append
(
sig
)
allsigv
.
append
(
sig
)
allsigv
.
sort
(
key
=
lambda
sig
:
sig
.
signo
)
allsigv
.
sort
(
key
=
lambda
sig
:
sig
.
signo
)
allsigv
.
remove
(
syscall
.
SIGKILL
)
# SIGKILL/SIGSTOP cannot be caught
def
without
(
signame
):
allsigv
.
remove
(
syscall
.
SIGSTOP
)
sig
=
getattr
(
syscall
,
signame
,
None
)
allsigv
.
remove
(
syscall
.
SIGBUS
)
# AddressSanitizer crashes on SIGBUS/SIGFPE/SIGSEGV
if
sig
is
not
None
:
allsigv
.
remove
(
syscall
.
SIGFPE
)
allsigv
.
remove
(
sig
)
allsigv
.
remove
(
syscall
.
SIGSEGV
)
without
(
'SIGKILL'
)
# SIGKILL/SIGSTOP cannot be caught
without
(
'SIGSTOP'
)
without
(
'SIGBUS'
)
# AddressSanitizer crashes on SIGBUS/SIGFPE/SIGSEGV
without
(
'SIGFPE'
)
without
(
'SIGSEGV'
)
without
(
'SIGILL'
)
# SIGILL/SIGABRT cause termination on windows
without
(
'SIGABRT'
)
# Notify() -> kill * -> should be notified
# Notify() -> kill * -> should be notified
ch
=
chan
(
10
,
dtype
=
gos
.
Signal
)
ch
=
chan
(
10
,
dtype
=
gos
.
Signal
)
...
@@ -72,11 +79,6 @@ def main():
...
@@ -72,11 +79,6 @@ def main():
killme
(
syscall
.
SIGTERM
)
killme
(
syscall
.
SIGTERM
)
raise
AssertionError
(
"not terminated"
)
raise
AssertionError
(
"not terminated"
)
# killme sends signal sig to own process.
def
killme
(
sig
):
mypid
=
os
.
getpid
()
os
.
kill
(
mypid
,
sig
.
signo
)
def
emit
(
msg
=
''
):
def
emit
(
msg
=
''
):
print
(
msg
)
print
(
msg
)
sys
.
stdout
.
flush
()
sys
.
stdout
.
flush
()
...
...
golang/pyx/build.py
View file @
2b56ce71
# Copyright (C) 2019-202
2
Nexedi SA and Contributors.
# Copyright (C) 2019-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -73,8 +73,10 @@ _dso_build_ext = setuptools_dso.build_ext
...
@@ -73,8 +73,10 @@ _dso_build_ext = setuptools_dso.build_ext
class
build_ext
(
_dso_build_ext
):
class
build_ext
(
_dso_build_ext
):
def
build_extension
(
self
,
ext
):
def
build_extension
(
self
,
ext
):
# wrap _compiler <src> -> <obj> with our code
# wrap _compiler <src> -> <obj> with our code
# ._compile is used on gcc/clang but not with msvc
_compile
=
self
.
compiler
.
_compile
_compile
=
self
.
compiler
.
_compile
def
_
(
obj
,
src
,
ext
,
cc_args
,
extra_postargs
,
pp_opts
):
def
xcompile
(
obj
,
src
,
ext
,
cc_args
,
extra_postargs
,
pp_opts
):
# filter_out removes arguments that start with argprefix
# filter_out removes arguments that start with argprefix
cc_args
=
cc_args
[:]
cc_args
=
cc_args
[:]
extra_postargs
=
extra_postargs
[:]
extra_postargs
=
extra_postargs
[:]
...
@@ -101,11 +103,30 @@ class build_ext(_dso_build_ext):
...
@@ -101,11 +103,30 @@ class build_ext(_dso_build_ext):
filter_out
(
'-std=gnu++'
)
filter_out
(
'-std=gnu++'
)
_compile
(
obj
,
src
,
ext
,
cc_args
,
extra_postargs
,
pp_opts
)
_compile
(
obj
,
src
,
ext
,
cc_args
,
extra_postargs
,
pp_opts
)
self
.
compiler
.
_compile
=
_
# msvc handles all sources directly in the loop in .compile and we can
# do per-source adjustsment only in .spawn .
spawn
=
self
.
compiler
.
spawn
def
xspawn
(
argv
):
c
=
False
for
arg
in
argv
:
if
arg
.
startswith
(
'/Tc'
):
c
=
True
if
c
:
argv
=
argv
[:]
for
i
in
range
(
len
(
argv
)):
if
argv
[
i
]
==
'/std:c++20'
:
argv
[
i
]
=
'/std:c11'
return
spawn
(
argv
)
self
.
compiler
.
_compile
=
xcompile
self
.
compiler
.
spawn
=
xspawn
try
:
try
:
_dso_build_ext
.
build_extension
(
self
,
ext
)
# super doesn't work for _dso_build_ext
_dso_build_ext
.
build_extension
(
self
,
ext
)
# super doesn't work for _dso_build_ext
finally
:
finally
:
self
.
compiler
.
_compile
=
_compile
self
.
compiler
.
_compile
=
_compile
self
.
compiler
.
spawn
=
spawn
# setup should be used instead of setuptools.setup
# setup should be used instead of setuptools.setup
...
@@ -128,14 +149,14 @@ def setup(**kw):
...
@@ -128,14 +149,14 @@ def setup(**kw):
# x_dsos = [DSO('mypkg.mydso', ['mypkg/mydso.cpp'])],
# x_dsos = [DSO('mypkg.mydso', ['mypkg/mydso.cpp'])],
# )
# )
def
DSO
(
name
,
sources
,
**
kw
):
def
DSO
(
name
,
sources
,
**
kw
):
_
,
kw
=
_with_build_defaults
(
kw
)
_
,
kw
=
_with_build_defaults
(
name
,
kw
)
dso
=
setuptools_dso
.
DSO
(
name
,
sources
,
**
kw
)
dso
=
setuptools_dso
.
DSO
(
name
,
sources
,
**
kw
)
return
dso
return
dso
# _with_build_defaults returns copy of kw amended with build options common for
# _with_build_defaults returns copy of kw amended with build options common for
# both DSO and Extension.
# both DSO and Extension.
def
_with_build_defaults
(
kw
):
# -> (pygo, kw')
def
_with_build_defaults
(
name
,
kw
):
# -> (pygo, kw')
# find pygolang root
# find pygolang root
gopkg
=
_findpkg
(
"golang"
)
gopkg
=
_findpkg
(
"golang"
)
pygo
=
dirname
(
gopkg
.
path
)
# .../pygolang/golang -> .../pygolang
pygo
=
dirname
(
gopkg
.
path
)
# .../pygolang/golang -> .../pygolang
...
@@ -144,15 +165,22 @@ def _with_build_defaults(kw): # -> (pygo, kw')
...
@@ -144,15 +165,22 @@ def _with_build_defaults(kw): # -> (pygo, kw')
kw
=
kw
.
copy
()
kw
=
kw
.
copy
()
# prepend -I<pygolang> so that e.g. golang/libgolang.h is found
sysname
=
platform
.
system
().
lower
()
win
=
(
sysname
==
'windows'
)
msvc
=
win
# TODO also support mingw ?
# prepend -I<pygolang> so that e.g. golang/libgolang.h is found
# same with -I<pygolang>/golang/_compat/<os>
incv
=
kw
.
get
(
'include_dirs'
,
[])[:]
incv
=
kw
.
get
(
'include_dirs'
,
[])[:]
incv
.
insert
(
0
,
pygo
)
incv
.
insert
(
0
,
pygo
)
incv
.
insert
(
1
,
join
(
pygo
,
'golang'
,
'_compat'
,
sysname
))
kw
[
'include_dirs'
]
=
incv
kw
[
'include_dirs'
]
=
incv
# link with libgolang.so
# link with libgolang.so if it is not libgolang itself
dsov
=
kw
.
get
(
'dsos'
,
[])[:]
if
name
!=
'golang.runtime.libgolang'
:
dsov
.
insert
(
0
,
'golang.runtime.libgolang'
)
dsov
=
kw
.
get
(
'dsos'
,
[])[:]
kw
[
'dsos'
]
=
dsov
dsov
.
insert
(
0
,
'golang.runtime.libgolang'
)
kw
[
'dsos'
]
=
dsov
# default language to C++ (chan[T] & co are accessible only via C++)
# default language to C++ (chan[T] & co are accessible only via C++)
lang
=
kw
.
setdefault
(
'language'
,
'c++'
)
lang
=
kw
.
setdefault
(
'language'
,
'c++'
)
...
@@ -160,9 +188,20 @@ def _with_build_defaults(kw): # -> (pygo, kw')
...
@@ -160,9 +188,20 @@ def _with_build_defaults(kw): # -> (pygo, kw')
# default to C++11 (chan[T] & co require C++11 features)
# default to C++11 (chan[T] & co require C++11 features)
ccdefault
=
[]
ccdefault
=
[]
if
lang
==
'c++'
:
if
lang
==
'c++'
:
ccdefault
.
append
(
'-std=c++11'
)
if
not
msvc
:
if
name
==
'golang.runtime.libgolang'
:
ccdefault
.
append
(
'-std=gnu++11'
)
# not c++11 as linux/list.h uses typeof
else
:
ccdefault
.
append
(
'-std=c++11'
)
else
:
# MSVC requries C++20 for designated struct initializers
ccdefault
.
append
(
'/std:c++20'
)
# and make exception handling: "fully conformant", so that e.g. extern "C" `panic` works
ccdefault
.
extend
([
'/EHc-'
,
'/EHsr'
])
# default to no strict-aliasing
# default to no strict-aliasing
ccdefault
.
append
(
'-fno-strict-aliasing'
)
if
not
msvc
:
ccdefault
.
append
(
'-fno-strict-aliasing'
)
# use only on gcc/clang - msvc does it by default
_
=
kw
.
get
(
'extra_compile_args'
,
[])[:]
_
=
kw
.
get
(
'extra_compile_args'
,
[])[:]
_
[
0
:
0
]
=
ccdefault
# if another e.g. -std=... was already there -
_
[
0
:
0
]
=
ccdefault
# if another e.g. -std=... was already there -
...
@@ -188,6 +227,8 @@ def _with_build_defaults(kw): # -> (pygo, kw')
...
@@ -188,6 +227,8 @@ def _with_build_defaults(kw): # -> (pygo, kw')
'os/signal.h'
,
'os/signal.h'
,
'pyx/runtime.h'
,
'pyx/runtime.h'
,
'_testing.h'
,
'_testing.h'
,
'_compat/windows/strings.h'
,
'_compat/windows/unistd.h'
,
]])
]])
kw
[
'depends'
]
=
dependv
kw
[
'depends'
]
=
dependv
...
@@ -203,7 +244,7 @@ def _with_build_defaults(kw): # -> (pygo, kw')
...
@@ -203,7 +244,7 @@ def _with_build_defaults(kw): # -> (pygo, kw')
# ext_modules = [Extension('mypkg.mymod', ['mypkg/mymod.pyx'])],
# ext_modules = [Extension('mypkg.mymod', ['mypkg/mymod.pyx'])],
# )
# )
def
Extension
(
name
,
sources
,
**
kw
):
def
Extension
(
name
,
sources
,
**
kw
):
pygo
,
kw
=
_with_build_defaults
(
kw
)
pygo
,
kw
=
_with_build_defaults
(
name
,
kw
)
# some pyx-level depends to workaround a bit lack of proper dependency
# some pyx-level depends to workaround a bit lack of proper dependency
# tracking in setuptools/distutils.
# tracking in setuptools/distutils.
...
...
golang/pyx/runtime.h
View file @
2b56ce71
#ifndef _NXD_LIBGOLANG_PYX_RUNTIME_H
#ifndef _NXD_LIBGOLANG_PYX_RUNTIME_H
#define _NXD_LIBGOLANG_PYX_RUNTIME_H
#define _NXD_LIBGOLANG_PYX_RUNTIME_H
// Copyright (C) 2018-20
19
Nexedi SA and Contributors.
// Copyright (C) 2018-20
23
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -58,8 +58,8 @@ class _PyError final : public _error, object {
...
@@ -58,8 +58,8 @@ class _PyError final : public _error, object {
private:
private:
_PyError
();
_PyError
();
~
_PyError
();
~
_PyError
();
friend
error
PyErr_Fetch
();
friend
LIBPYXRUNTIME_API
error
PyErr_Fetch
();
friend
void
PyErr_ReRaise
(
PyError
pyerr
);
friend
LIBPYXRUNTIME_API
void
PyErr_ReRaise
(
PyError
pyerr
);
public:
public:
LIBPYXRUNTIME_API
void
incref
();
LIBPYXRUNTIME_API
void
incref
();
LIBPYXRUNTIME_API
void
decref
();
LIBPYXRUNTIME_API
void
decref
();
...
...
golang/pyx/testprog/cmdclass_custom.py
View file @
2b56ce71
# Copyright (C) 2019 Nexedi SA and Contributors.
# Copyright (C) 2019
-2022
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
#
Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
# it under the terms of the GNU General Public License version 3, or (at your
# it under the terms of the GNU General Public License version 3, or (at your
...
@@ -29,4 +29,9 @@ class mybuild_ext(build_ext):
...
@@ -29,4 +29,9 @@ class mybuild_ext(build_ext):
setup
(
setup
(
cmdclass
=
{
'build_ext'
:
mybuild_ext
},
cmdclass
=
{
'build_ext'
:
mybuild_ext
},
# avoid setuptools thinking nearby golang_pyx_user/ golang_dso_user/ also
# relate to hereby setup and rejecting the build.
# https://stackoverflow.com/questions/72294299/multiple-top-level-packages-discovered-in-a-flat-layout
py_modules
=
[],
)
)
golang/pyx/testprog/golang_dso_user/dsouser/dso.cpp
View file @
2b56ce71
// Copyright (C) 2019 Nexedi SA and Contributors.
// Copyright (C) 2019
-2023
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
//
Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
// it under the terms of the GNU General Public License version 3, or (at your
// it under the terms of the GNU General Public License version 3, or (at your
...
@@ -20,7 +20,7 @@
...
@@ -20,7 +20,7 @@
// Small library that uses a bit of libgolang features, mainly to verify
// Small library that uses a bit of libgolang features, mainly to verify
// that external project can build against libgolang.
// that external project can build against libgolang.
#include
<golang/libgolang.h>
#include
"dso.h"
using
namespace
golang
;
using
namespace
golang
;
#include <stdio.h>
#include <stdio.h>
...
...
golang/pyx/testprog/golang_dso_user/dsouser/dso.h
0 → 100644
View file @
2b56ce71
// Copyright (C) 2023 Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
//
// This program is free software: you can Use, Study, Modify and Redistribute
// it under the terms of the GNU General Public License version 3, or (at your
// option) any later version, as published by the Free Software Foundation.
//
// You can also Link and Combine this program with other software covered by
// the terms of any of the Free Software licenses or any of the Open Source
// Initiative approved licenses and Convey the resulting work. Corresponding
// source of such a combination shall include the source code for all other
// software used.
//
// This program is distributed WITHOUT ANY WARRANTY; without even the implied
// warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
//
// See COPYING file for full licensing terms.
// See https://www.nexedi.com/licensing for rationale and options.
#ifndef _NXD_GOLANG_DSOUSER_DSO_H
#define _NXD_GOLANG_DSOUSER_DSO_H
#include <golang/libgolang.h>
#if BUILDING_DSOUSER_DSO
# define DSOUSER_DSO_API LIBGOLANG_DSO_EXPORT
#else
# define DSOUSER_DSO_API LIBGOLANG_DSO_IMPORT
#endif
DSOUSER_DSO_API
void
dsotest
();
#endif // _NXD_GOLANG_DSOUSER_DSO_H
golang/pyx/testprog/golang_dso_user/dsouser/test.pyx
View file @
2b56ce71
# cython: language_level=2
# cython: language_level=2
#
#
# Copyright (C) 2019 Nexedi SA and Contributors.
# Copyright (C) 2019
-2023
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
#
Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
# it under the terms of the GNU General Public License version 3, or (at your
# it under the terms of the GNU General Public License version 3, or (at your
...
@@ -21,10 +21,7 @@
...
@@ -21,10 +21,7 @@
# Small test driver that calls dso.so .
# Small test driver that calls dso.so .
cdef
extern
from
*
nogil
:
cdef
extern
from
"dso.h"
nogil
:
"""
void dsotest();
"""
void
dsotest
()
void
dsotest
()
def
main
():
def
main
():
...
...
golang/pyx/testprog/golang_dso_user/setup.py
View file @
2b56ce71
# Copyright (C) 2019 Nexedi SA and Contributors.
# Copyright (C) 2019
-2023
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
#
Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
# it under the terms of the GNU General Public License version 3, or (at your
# it under the terms of the GNU General Public License version 3, or (at your
...
@@ -24,7 +24,9 @@ setup(
...
@@ -24,7 +24,9 @@ setup(
name
=
'golang_dso_user'
,
name
=
'golang_dso_user'
,
description
=
'test project that uses libgolang from a dso'
,
description
=
'test project that uses libgolang from a dso'
,
x_dsos
=
[
DSO
(
'dsouser.dso'
,
[
'dsouser/dso.cpp'
])],
x_dsos
=
[
DSO
(
'dsouser.dso'
,
[
'dsouser/dso.cpp'
],
define_macros
=
[(
'BUILDING_DSOUSER_DSO'
,
None
)])],
ext_modules
=
[
Extension
(
'dsouser.test'
,
ext_modules
=
[
Extension
(
'dsouser.test'
,
[
'dsouser/test.pyx'
],
[
'dsouser/test.pyx'
],
dsos
=
[
'dsouser.dso'
])],
dsos
=
[
'dsouser.dso'
])],
...
...
golang/runtime/_runtime_gevent.pyx
View file @
2b56ce71
# cython: language_level=2
# cython: language_level=2
# Copyright (C) 2019-202
2
Nexedi SA and Contributors.
# Copyright (C) 2019-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -57,7 +57,7 @@ from posix.fcntl cimport mode_t, F_GETFL, F_SETFL, O_NONBLOCK, O_ACCMODE, O_RDON
...
@@ -57,7 +57,7 @@ from posix.fcntl cimport mode_t, F_GETFL, F_SETFL, O_NONBLOCK, O_ACCMODE, O_RDON
from
posix.stat
cimport
struct_stat
,
S_ISREG
,
S_ISDIR
,
S_ISBLK
from
posix.stat
cimport
struct_stat
,
S_ISREG
,
S_ISDIR
,
S_ISBLK
from
posix.strings
cimport
bzero
from
posix.strings
cimport
bzero
from
gevent
.fileobject
import
FileObjectThread
,
FileObjectPosix
from
gevent
import
fileobject
as
gfileobj
# _goviapy & _togo serve go
# _goviapy & _togo serve go
...
@@ -194,31 +194,39 @@ cdef nogil:
...
@@ -194,31 +194,39 @@ cdef nogil:
return
ioh
return
ioh
_libgolang_ioh
*
_io_fdopen
(
int
*
out_syserr
,
int
sysfd
):
_libgolang_ioh
*
_io_fdopen
(
int
*
out_syserr
,
int
sysfd
):
# check if we should enable O_NONBLOCK on this file-descriptor
IF
POSIX
:
# even though we could enable O_NONBLOCK for regular files, it does not
# check if we should enable O_NONBLOCK on this file-descriptor
# work as expected as most unix'es report regular files as always read
# even though we could enable O_NONBLOCK for regular files, it does not
# and write ready.
# work as expected as most unix'es report regular files as always read
cdef
struct_stat
st
# and write ready.
cdef
int
syserr
=
syscall
.
Fstat
(
sysfd
,
&
st
)
cdef
struct_stat
st
if
syserr
<
0
:
cdef
int
syserr
=
syscall
.
Fstat
(
sysfd
,
&
st
)
out_syserr
[
0
]
=
syserr
return
NULL
m
=
st
.
st_mode
blocking
=
(
S_ISREG
(
m
)
or
S_ISDIR
(
m
)
or
S_ISBLK
(
m
))
# fd cannot refer to symlink
# retrieve current sysfd flags and access mode
flags
=
syscall
.
Fcntl
(
sysfd
,
F_GETFL
,
0
)
if
flags
<
0
:
out_syserr
[
0
]
=
flags
return
NULL
acc
=
(
flags
&
O_ACCMODE
)
# enable O_NONBLOCK if needed
if
not
blocking
:
syserr
=
syscall
.
Fcntl
(
sysfd
,
F_SETFL
,
flags
|
O_NONBLOCK
)
if
syserr
<
0
:
if
syserr
<
0
:
out_syserr
[
0
]
=
syserr
out_syserr
[
0
]
=
syserr
return
NULL
return
NULL
m
=
st
.
st_mode
blocking
=
(
S_ISREG
(
m
)
or
S_ISDIR
(
m
)
or
S_ISBLK
(
m
))
# fd cannot refer to symlink
# retrieve current sysfd flags and access mode
flags
=
syscall
.
Fcntl
(
sysfd
,
F_GETFL
,
0
)
if
flags
<
0
:
out_syserr
[
0
]
=
flags
return
NULL
acc
=
(
flags
&
O_ACCMODE
)
# enable O_NONBLOCK if needed
if
not
blocking
:
syserr
=
syscall
.
Fcntl
(
sysfd
,
F_SETFL
,
flags
|
O_NONBLOCK
)
if
syserr
<
0
:
out_syserr
[
0
]
=
syserr
return
NULL
ELSE
:
# !POSIX (windows)
cdef
bint
blocking
=
True
# FIXME acc: use GetFileInformationByHandleEx to determine whether
# HANDLE(sysfd) was opened for reading or writing.
# https://stackoverflow.com/q/9442436/9456786
cdef
int
acc
=
O_RDWR
# create IOH backed by FileObjectThread or FileObjectPosix
# create IOH backed by FileObjectThread or FileObjectPosix
ioh
=
<
IOH
*>
calloc
(
1
,
sizeof
(
IOH
))
ioh
=
<
IOH
*>
calloc
(
1
,
sizeof
(
IOH
))
...
@@ -253,9 +261,9 @@ cdef:
...
@@ -253,9 +261,9 @@ cdef:
pygfobj
=
None
pygfobj
=
None
try
:
try
:
if
blocking
:
if
blocking
:
pygfobj
=
FileObjectThread
(
sysfd
,
mode
=
mode
,
buffering
=
0
)
pygfobj
=
gfileobj
.
FileObjectThread
(
sysfd
,
mode
=
mode
,
buffering
=
0
)
else
:
else
:
pygfobj
=
FileObjectPosix
(
sysfd
,
mode
=
mode
,
buffering
=
0
)
pygfobj
=
gfileobj
.
FileObjectPosix
(
sysfd
,
mode
=
mode
,
buffering
=
0
)
except
OSError
as
e
:
except
OSError
as
e
:
out_syserr
[
0
]
=
-
e
.
errno
out_syserr
[
0
]
=
-
e
.
errno
else
:
else
:
...
@@ -338,7 +346,19 @@ cdef:
...
@@ -338,7 +346,19 @@ cdef:
cdef
uint8_t
[::
1
]
mem
=
<
uint8_t
[:
count
]
>
buf
cdef
uint8_t
[::
1
]
mem
=
<
uint8_t
[:
count
]
>
buf
xmem
=
memoryview
(
mem
)
# to avoid https://github.com/cython/cython/issues/3900 on mem[:0]=b''
xmem
=
memoryview
(
mem
)
# to avoid https://github.com/cython/cython/issues/3900 on mem[:0]=b''
try
:
try
:
n
=
pygfobj
.
readinto
(
xmem
)
# NOTE buf might be on stack, so it must not be accessed, e.g. from
# FileObjectThread, while our greenlet is parked (see STACK_DEAD_WHILE_PARKED
# for details). -> Use intermediate on-heap buffer to protect from that.
#
# Also: we cannot use pygfobj.readinto due to
# https://github.com/gevent/gevent/pull/1948
#
# TODO use .readinto() when buf is not on stack and gevent is
# recent enough or pygfobj is not FileObjectThread.
#n = pygfobj.readinto(xmem)
buf2
=
pygfobj
.
read
(
count
)
n
=
len
(
buf2
)
xmem
[:
n
]
=
buf2
except
OSError
as
e
:
except
OSError
as
e
:
n
=
-
e
.
errno
n
=
-
e
.
errno
out_n
[
0
]
=
n
out_n
[
0
]
=
n
...
@@ -361,8 +381,14 @@ cdef:
...
@@ -361,8 +381,14 @@ cdef:
bint
_io_write
(
IOH
*
ioh
,
int
*
out_n
,
const
void
*
buf
,
size_t
count
):
bint
_io_write
(
IOH
*
ioh
,
int
*
out_n
,
const
void
*
buf
,
size_t
count
):
pygfobj
=
<
object
>
ioh
.
pygfobj
pygfobj
=
<
object
>
ioh
.
pygfobj
cdef
const
uint8_t
[::
1
]
mem
=
<
const
uint8_t
[:
count
]
>
buf
cdef
const
uint8_t
[::
1
]
mem
=
<
const
uint8_t
[:
count
]
>
buf
# NOTE buf might be on stack, so it must not be accessed, e.g. from
# FileObjectThread, while our greenlet is parked (see STACK_DEAD_WHILE_PARKED
# for details). -> Use intermediate on-heap buffer to protect from that.
buf2
=
bytearray
(
mem
)
try
:
try
:
n
=
pygfobj
.
write
(
mem
)
n
=
pygfobj
.
write
(
buf2
)
except
OSError
as
e
:
except
OSError
as
e
:
n
=
-
e
.
errno
n
=
-
e
.
errno
out_n
[
0
]
=
n
out_n
[
0
]
=
n
...
...
golang/runtime/internal/atomic.cpp
View file @
2b56ce71
// Copyright (C) 2022 Nexedi SA and Contributors.
// Copyright (C) 2022
-2023
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
//
Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
// it under the terms of the GNU General Public License version 3, or (at your
// it under the terms of the GNU General Public License version 3, or (at your
...
@@ -20,7 +20,9 @@
...
@@ -20,7 +20,9 @@
#include "golang/runtime/internal/atomic.h"
#include "golang/runtime/internal/atomic.h"
#include "golang/libgolang.h"
#include "golang/libgolang.h"
#ifndef _WIN32
#include <pthread.h>
#include <pthread.h>
#endif
// golang::internal::atomic::
// golang::internal::atomic::
namespace
golang
{
namespace
golang
{
...
@@ -41,9 +43,12 @@ static void _forkNewEpoch() {
...
@@ -41,9 +43,12 @@ static void _forkNewEpoch() {
}
}
void
_init
()
{
void
_init
()
{
// there is no fork on windows
#ifndef _WIN32
int
e
=
pthread_atfork
(
/*prepare*/
nil
,
/*inparent*/
nil
,
/*inchild*/
_forkNewEpoch
);
int
e
=
pthread_atfork
(
/*prepare*/
nil
,
/*inparent*/
nil
,
/*inchild*/
_forkNewEpoch
);
if
(
e
!=
0
)
if
(
e
!=
0
)
panic
(
"pthread_atfork failed"
);
panic
(
"pthread_atfork failed"
);
#endif
}
}
...
...
golang/runtime/internal/syscall.cpp
View file @
2b56ce71
// Copyright (C) 2021-202
2
Nexedi SA and Contributors.
// Copyright (C) 2021-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -57,15 +57,18 @@ string _Errno::Error() {
...
@@ -57,15 +57,18 @@ string _Errno::Error() {
_Errno
&
e
=
*
this
;
_Errno
&
e
=
*
this
;
char
ebuf
[
128
];
char
ebuf
[
128
];
bool
ok
;
#if __APPLE__
#if __APPLE__
int
x
=
::
strerror_r
(
-
e
.
syserr
,
ebuf
,
sizeof
(
ebuf
));
ok
=
(
::
strerror_r
(
-
e
.
syserr
,
ebuf
,
sizeof
(
ebuf
))
==
0
);
if
(
x
==
0
)
#elif defined(_WIN32)
return
string
(
ebuf
);
ok
=
(
::
strerror_s
(
ebuf
,
sizeof
(
ebuf
),
-
e
.
syserr
)
==
0
);
return
"unknown error "
+
std
::
to_string
(
-
e
.
syserr
);
#else
#else
char
*
estr
=
::
strerror_r
(
-
e
.
syserr
,
ebuf
,
sizeof
(
ebuf
));
char
*
estr
=
::
strerror_r
(
-
e
.
syserr
,
ebuf
,
sizeof
(
ebuf
));
return
string
(
estr
);
return
string
(
estr
);
#endif
#endif
if
(
ok
)
return
string
(
ebuf
);
return
"unknown error "
+
std
::
to_string
(
-
e
.
syserr
);
}
}
...
@@ -99,6 +102,7 @@ __Errno Close(int fd) {
...
@@ -99,6 +102,7 @@ __Errno Close(int fd) {
return
err
;
return
err
;
}
}
#ifndef _WIN32
__Errno
Fcntl
(
int
fd
,
int
cmd
,
int
arg
)
{
__Errno
Fcntl
(
int
fd
,
int
cmd
,
int
arg
)
{
int
save_errno
=
errno
;
int
save_errno
=
errno
;
int
err
=
::
fcntl
(
fd
,
cmd
,
arg
);
int
err
=
::
fcntl
(
fd
,
cmd
,
arg
);
...
@@ -107,6 +111,7 @@ __Errno Fcntl(int fd, int cmd, int arg) {
...
@@ -107,6 +111,7 @@ __Errno Fcntl(int fd, int cmd, int arg) {
errno
=
save_errno
;
errno
=
save_errno
;
return
err
;
return
err
;
}
}
#endif
__Errno
Fstat
(
int
fd
,
struct
::
stat
*
out_st
)
{
__Errno
Fstat
(
int
fd
,
struct
::
stat
*
out_st
)
{
int
save_errno
=
errno
;
int
save_errno
=
errno
;
...
@@ -119,6 +124,10 @@ __Errno Fstat(int fd, struct ::stat *out_st) {
...
@@ -119,6 +124,10 @@ __Errno Fstat(int fd, struct ::stat *out_st) {
int
Open
(
const
char
*
path
,
int
flags
,
mode_t
mode
)
{
int
Open
(
const
char
*
path
,
int
flags
,
mode_t
mode
)
{
int
save_errno
=
errno
;
int
save_errno
=
errno
;
#ifdef _WIN32 // default to open files in binary mode
if
((
flags
&
(
_O_TEXT
|
_O_BINARY
))
==
0
)
flags
|=
_O_BINARY
;
#endif
int
fd
=
::
open
(
path
,
flags
,
mode
);
int
fd
=
::
open
(
path
,
flags
,
mode
);
if
(
fd
<
0
)
if
(
fd
<
0
)
fd
=
-
errno
;
fd
=
-
errno
;
...
@@ -126,15 +135,39 @@ int Open(const char *path, int flags, mode_t mode) {
...
@@ -126,15 +135,39 @@ int Open(const char *path, int flags, mode_t mode) {
return
fd
;
return
fd
;
}
}
__Errno
Pipe
(
int
vfd
[
2
])
{
__Errno
Pipe
(
int
vfd
[
2
],
int
flags
)
{
// supported flags: O_CLOEXEC
if
(
flags
&
~
(
O_CLOEXEC
))
return
-
EINVAL
;
int
save_errno
=
errno
;
int
save_errno
=
errno
;
int
err
=
::
pipe
(
vfd
);
int
err
;
#ifdef __linux__
err
=
::
pipe2
(
vfd
,
flags
);
#elif defined(_WIN32)
err
=
::
_pipe
(
vfd
,
4096
,
flags
|
_O_BINARY
);
#else
err
=
::
pipe
(
vfd
);
if
(
err
)
goto
out
;
if
(
flags
&
O_CLOEXEC
)
{
for
(
int
i
=
0
;
i
<
2
;
i
++
)
{
err
=
Fcntl
(
vfd
[
i
],
F_SETFD
,
FD_CLOEXEC
);
if
(
err
)
{
Close
(
vfd
[
0
]);
Close
(
vfd
[
1
]);
goto
out
;
}
}
}
out:
#endif
if
(
err
==
-
1
)
if
(
err
==
-
1
)
err
=
-
errno
;
err
=
-
errno
;
errno
=
save_errno
;
errno
=
save_errno
;
return
err
;
return
err
;
}
}
#ifndef _WIN32
__Errno
Sigaction
(
int
signo
,
const
struct
::
sigaction
*
act
,
struct
::
sigaction
*
oldact
)
{
__Errno
Sigaction
(
int
signo
,
const
struct
::
sigaction
*
act
,
struct
::
sigaction
*
oldact
)
{
int
save_errno
=
errno
;
int
save_errno
=
errno
;
int
err
=
::
sigaction
(
signo
,
act
,
oldact
);
int
err
=
::
sigaction
(
signo
,
act
,
oldact
);
...
@@ -143,6 +176,17 @@ __Errno Sigaction(int signo, const struct ::sigaction *act, struct ::sigaction *
...
@@ -143,6 +176,17 @@ __Errno Sigaction(int signo, const struct ::sigaction *act, struct ::sigaction *
errno
=
save_errno
;
errno
=
save_errno
;
return
err
;
return
err
;
}
}
#endif
sighandler_t
Signal
(
int
signo
,
sighandler_t
handler
,
int
*
out_psyserr
)
{
int
save_errno
=
errno
;
sighandler_t
oldh
=
::
signal
(
signo
,
handler
);
*
out_psyserr
=
(
oldh
==
SIG_ERR
?
(
errno
?
-
errno
:
-
EINVAL
)
// windows returns SIG_ERR/errno=0 for invalid signo
:
0
);
errno
=
save_errno
;
return
oldh
;
}
}}}
// golang::internal::syscall::
}}}
// golang::internal::syscall::
golang/runtime/internal/syscall.h
View file @
2b56ce71
#ifndef _NXD_LIBGOLANG_RUNTIME_INTERNAL_SYSCALL_H
#ifndef _NXD_LIBGOLANG_RUNTIME_INTERNAL_SYSCALL_H
#define _NXD_LIBGOLANG_RUNTIME_INTERNAL_SYSCALL_H
#define _NXD_LIBGOLANG_RUNTIME_INTERNAL_SYSCALL_H
// Copyright (C) 2021-202
2
Nexedi SA and Contributors.
// Copyright (C) 2021-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -59,15 +59,21 @@ error NewErrno(__Errno syserr); // TODO better return Errno directly.
...
@@ -59,15 +59,21 @@ error NewErrno(__Errno syserr); // TODO better return Errno directly.
// system calls
// system calls
int
/*n|err*/
Read
(
int
fd
,
void
*
buf
,
size_t
count
);
LIBGOLANG_API
int
/*n|err*/
Read
(
int
fd
,
void
*
buf
,
size_t
count
);
int
/*n|err*/
Write
(
int
fd
,
const
void
*
buf
,
size_t
count
);
LIBGOLANG_API
int
/*n|err*/
Write
(
int
fd
,
const
void
*
buf
,
size_t
count
);
__Errno
Close
(
int
fd
);
LIBGOLANG_API
__Errno
Close
(
int
fd
);
__Errno
Fcntl
(
int
fd
,
int
cmd
,
int
arg
);
#ifndef _WIN32
__Errno
Fstat
(
int
fd
,
struct
::
stat
*
out_st
);
LIBGOLANG_API
__Errno
Fcntl
(
int
fd
,
int
cmd
,
int
arg
);
int
/*fd|err*/
Open
(
const
char
*
path
,
int
flags
,
mode_t
mode
);
#endif
__Errno
Pipe
(
int
vfd
[
2
]);
LIBGOLANG_API
__Errno
Fstat
(
int
fd
,
struct
::
stat
*
out_st
);
__Errno
Sigaction
(
int
signo
,
const
struct
::
sigaction
*
act
,
struct
::
sigaction
*
oldact
);
LIBGOLANG_API
int
/*fd|err*/
Open
(
const
char
*
path
,
int
flags
,
mode_t
mode
);
LIBGOLANG_API
__Errno
Pipe
(
int
vfd
[
2
],
int
flags
);
#ifndef _WIN32
LIBGOLANG_API
__Errno
Sigaction
(
int
signo
,
const
struct
::
sigaction
*
act
,
struct
::
sigaction
*
oldact
);
#endif
typedef
void
(
*
sighandler_t
)(
int
);
LIBGOLANG_API
sighandler_t
/*sigh|SIG_ERR*/
Signal
(
int
signo
,
sighandler_t
handler
,
int
*
out_psyserr
);
}}}
// golang::internal::syscall::
}}}
// golang::internal::syscall::
...
...
golang/runtime/libgolang.cpp
View file @
2b56ce71
// Copyright (C) 2018-202
2
Nexedi SA and Contributors.
// Copyright (C) 2018-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -43,12 +43,23 @@
...
@@ -43,12 +43,23 @@
#include <stdlib.h>
#include <stdlib.h>
#include <string.h>
#include <string.h>
#include <strings.h>
// linux/list.h needs ARRAY_SIZE XXX -> better use c.h or ccan/array_size.h ?
// linux/list.h needs ARRAY_SIZE XXX -> better use c.h or ccan/array_size.h ?
#ifndef ARRAY_SIZE
#ifndef ARRAY_SIZE
# define ARRAY_SIZE(A) (sizeof(A) / sizeof((A)[0]))
# define ARRAY_SIZE(A) (sizeof(A) / sizeof((A)[0]))
#endif
#endif
#include <linux/list.h>
#include <linux/list.h>
// MSVC does not support statement expressions and typeof
// -> redo list_entry via C++ lambda.
#ifdef _MSC_VER
# undef list_entry
# define list_entry(ptr, type, member) [&]() { \
const decltype( ((type *)0)->member ) *__mptr = (ptr); \
return (type *)( (char *)__mptr - offsetof(type,member) ); \
}()
#endif
using
std
::
atomic
;
using
std
::
atomic
;
using
std
::
bad_alloc
;
using
std
::
bad_alloc
;
...
@@ -899,6 +910,13 @@ template<> int _chanselect2</*onstack=*/true> (const _selcase *, int, const vect
...
@@ -899,6 +910,13 @@ template<> int _chanselect2</*onstack=*/true> (const _selcase *, int, const vect
template
<
>
int
_chanselect2
<
/*onstack=*/
false
>
(
const
_selcase
*
,
int
,
const
vector
<
int
>&
);
template
<
>
int
_chanselect2
<
/*onstack=*/
false
>
(
const
_selcase
*
,
int
,
const
vector
<
int
>&
);
static
int
__chanselect2
(
const
_selcase
*
,
int
,
const
vector
<
int
>&
,
_WaitGroup
*
);
static
int
__chanselect2
(
const
_selcase
*
,
int
,
const
vector
<
int
>&
,
_WaitGroup
*
);
// PRNG for select.
// TODO consider switching to xoroshiro or wyrand if speed is an issue
// https://thompsonsed.co.uk/random-number-generators-for-c-performance-tested
// https://nullprogram.com/blog/2017/09/21/
static
std
::
random_device
_devrand
;
static
thread_local
std
::
mt19937
_t_rng
(
_devrand
());
// _chanselect executes one ready send or receive channel case.
// _chanselect executes one ready send or receive channel case.
//
//
// if no case is ready and default case was provided, select chooses default.
// if no case is ready and default case was provided, select chooses default.
...
@@ -925,7 +943,7 @@ int _chanselect(const _selcase *casev, int casec) {
...
@@ -925,7 +943,7 @@ int _chanselect(const _selcase *casev, int casec) {
vector
<
int
>
nv
(
casec
);
// n -> n(case) TODO -> caller stack-allocate nv
vector
<
int
>
nv
(
casec
);
// n -> n(case) TODO -> caller stack-allocate nv
for
(
int
i
=
0
;
i
<
casec
;
i
++
)
for
(
int
i
=
0
;
i
<
casec
;
i
++
)
nv
[
i
]
=
i
;
nv
[
i
]
=
i
;
std
::
random_shuffle
(
nv
.
begin
(),
nv
.
end
()
);
std
::
shuffle
(
nv
.
begin
(),
nv
.
end
(),
_t_rng
);
// first pass: poll all cases and bail out in the end if default was provided
// first pass: poll all cases and bail out in the end if default was provided
int
ndefault
=
-
1
;
int
ndefault
=
-
1
;
...
...
golang/sync.h
View file @
2b56ce71
#ifndef _NXD_LIBGOLANG_SYNC_H
#ifndef _NXD_LIBGOLANG_SYNC_H
#define _NXD_LIBGOLANG_SYNC_H
#define _NXD_LIBGOLANG_SYNC_H
// Copyright (C) 2018-202
0
Nexedi SA and Contributors.
// Copyright (C) 2018-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -212,7 +212,7 @@ class _WorkGroup : public object {
...
@@ -212,7 +212,7 @@ class _WorkGroup : public object {
private:
private:
_WorkGroup
();
_WorkGroup
();
~
_WorkGroup
();
~
_WorkGroup
();
friend
WorkGroup
NewWorkGroup
(
context
::
Context
ctx
);
friend
LIBGOLANG_API
WorkGroup
NewWorkGroup
(
context
::
Context
ctx
);
public:
public:
LIBGOLANG_API
void
decref
();
LIBGOLANG_API
void
decref
();
...
...
golang/testprog/golang_test_defer_excchain.txt
View file @
2b56ce71
Traceback (most recent call last):
Traceback (most recent call last):
File "PYGOLANG/golang/__init__.py", line ..., in _
File "PYGOLANG/golang/__init__.py", line ..., in _
return f(*argv, **kw)
return f(*argv, **kw)
^^^^^^^^^^^^^^ +PY311
File "PY39(PYGOLANG/golang/testprog/)golang_test_defer_excchain.py", line 42, in main
File "PY39(PYGOLANG/golang/testprog/)golang_test_defer_excchain.py", line 42, in main
raise RuntimeError("err")
raise RuntimeError("err")
RuntimeError: err
RuntimeError: err
...
@@ -21,6 +22,7 @@ Traceback (most recent call last):
...
@@ -21,6 +22,7 @@ Traceback (most recent call last):
...
...
File "PY39(PYGOLANG/golang/testprog/)golang_test_defer_excchain.py", line 33, in d2
File "PY39(PYGOLANG/golang/testprog/)golang_test_defer_excchain.py", line 33, in d2
1/0
1/0
~^~ +PY311
ZeroDivisionError: ...
ZeroDivisionError: ...
During handling of the above exception, another exception occurred:
During handling of the above exception, another exception occurred:
...
@@ -30,7 +32,8 @@ Traceback (most recent call last):
...
@@ -30,7 +32,8 @@ Traceback (most recent call last):
main()
main()
...
...
File "PYGOLANG/golang/__init__.py", line ..., in _
File "PYGOLANG/golang/__init__.py", line ..., in _
return f(*argv, **kw)
return f(*argv, **kw) -PY310
with __goframe__: +PY310
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
...
...
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
File "PYGOLANG/golang/__init__.py", line ..., in __exit__
...
...
golang/testprog/golang_test_defer_excchain.txt-pytest
View file @
2b56ce71
...
...
_____________________________________ main _____________________________________
...________________ main ________________...
../__init__.py:...: in _
../__init__.py:...: in _
return f(*argv, **kw)
return f(*argv, **kw)
golang_test_defer_excchain.py:42: in main
golang_test_defer_excchain.py:42: in main
...
...
golang/testprog/golang_test_str.py
View file @
2b56ce71
#!/usr/bin/env python
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
# Copyright (C) 2022 Nexedi SA and Contributors.
# Copyright (C) 2022
-2023
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
#
Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
# it under the terms of the GNU General Public License version 3, or (at your
# it under the terms of the GNU General Public License version 3, or (at your
...
@@ -29,8 +29,8 @@ from golang import b, u
...
@@ -29,8 +29,8 @@ from golang import b, u
from
golang.gcompat
import
qq
from
golang.gcompat
import
qq
def
main
():
def
main
():
sb
=
b
(
"привет b"
)
sb
=
b
(
"привет
αβγ
b"
)
su
=
u
(
"привет u"
)
su
=
u
(
"привет
αβγ
u"
)
print
(
"print(b):"
,
sb
)
print
(
"print(b):"
,
sb
)
print
(
"print(u):"
,
su
)
print
(
"print(u):"
,
su
)
print
(
"print(qq(b)):"
,
qq
(
sb
))
print
(
"print(qq(b)):"
,
qq
(
sb
))
...
...
golang/testprog/golang_test_str.txt
View file @
2b56ce71
print(b): привет b
print(b): привет
αβγ
b
print(u): привет u
print(u): привет
αβγ
u
print(qq(b)): "привет b"
print(qq(b)): "привет
αβγ
b"
print(qq(u)): "привет u"
print(qq(u)): "привет
αβγ
u"
print(repr(b)): b('привет b')
print(repr(b)): b('привет
αβγ
b')
print(repr(u)): u('привет u')
print(repr(u)): u('привет
αβγ
u')
print({b: u}): {b('привет
b'): u('привет
u')}
print({b: u}): {b('привет
αβγ b'): u('привет αβγ
u')}
golang/time.h
View file @
2b56ce71
#ifndef _NXD_LIBGOLANG_TIME_H
#ifndef _NXD_LIBGOLANG_TIME_H
#define _NXD_LIBGOLANG_TIME_H
#define _NXD_LIBGOLANG_TIME_H
// Copyright (C) 2019-202
0
Nexedi SA and Contributors.
// Copyright (C) 2019-202
3
Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
// Kirill Smelkov <kirr@nexedi.com>
//
//
// This program is free software: you can Use, Study, Modify and Redistribute
// This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -123,7 +123,7 @@ private:
...
@@ -123,7 +123,7 @@ private:
private:
private:
_Ticker
();
_Ticker
();
~
_Ticker
();
~
_Ticker
();
friend
Ticker
new_ticker
(
double
dt
);
friend
Ticker
LIBGOLANG_API
new_ticker
(
double
dt
);
public:
public:
LIBGOLANG_API
void
decref
();
LIBGOLANG_API
void
decref
();
...
...
gpython/__init__.py
View file @
2b56ce71
#!/usr/bin/env python
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
# Copyright (C) 2018-202
1
Nexedi SA and Contributors.
# Copyright (C) 2018-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -50,19 +50,27 @@ _pyopt_long = ('version',)
...
@@ -50,19 +50,27 @@ _pyopt_long = ('version',)
# init, if provided, is called after options are parsed, but before interpreter start.
# init, if provided, is called after options are parsed, but before interpreter start.
def
pymain
(
argv
,
init
=
None
):
def
pymain
(
argv
,
init
=
None
):
import
sys
import
sys
from
os.path
import
dirname
,
realpath
import
os
from
os.path
import
dirname
,
realpath
,
splitext
# sys.executable
# sys.executable
# on windows there are
# on windows there are
# gpython-script.py
# gpython.exe
# gpython.exe
# gpython-script.py
# gpython.manifest
# gpython.manifest
# and argv[0] is gpython-script.py
# with gpython-script.py sometimes embedded into gpython.exe (pip/distlib)
# and argv[0] is 'gpython-script.py' (setuptools) or 'gpython' (pip/distlib; note no .exe)
exe
=
realpath
(
argv
[
0
])
exe
=
realpath
(
argv
[
0
])
argv
=
argv
[
1
:]
argv
=
argv
[
1
:]
if
exe
.
endswith
(
'-script.py'
):
if
os
.
name
==
'nt'
:
exe
=
exe
[:
-
len
(
'-script.py'
)]
if
exe
.
endswith
(
'-script.py'
):
exe
=
exe
+
'.exe'
exe
=
exe
[:
-
len
(
'-script.py'
)]
# gpython-script.py -> gpython.exe
exe
=
exe
+
'.exe'
else
:
_
,
ext
=
splitext
(
exe
)
# gpython -> gpython.exe
if
not
ext
:
exe
+=
'.exe'
sys
.
_gpy_underlying_executable
=
sys
.
executable
sys
.
_gpy_underlying_executable
=
sys
.
executable
sys
.
executable
=
exe
sys
.
executable
=
exe
...
@@ -70,20 +78,26 @@ def pymain(argv, init=None):
...
@@ -70,20 +78,26 @@ def pymain(argv, init=None):
# `gpython file` will add path-to-file to sys.path[0] by itself, and
# `gpython file` will add path-to-file to sys.path[0] by itself, and
# /path/to/gpython is unnecessary and would create difference in behaviour
# /path/to/gpython is unnecessary and would create difference in behaviour
# in between gpython and python.
# in between gpython and python.
#
# on windows when gpython.exe comes with embedded __main__.py, it is
# gpython.exe that is installed into sys.path[0] .
exedir
=
dirname
(
exe
)
exedir
=
dirname
(
exe
)
if
sys
.
path
[
0
]
==
exedir
:
if
sys
.
path
[
0
]
in
(
exedir
,
exe
)
:
del
sys
.
path
[
0
]
del
sys
.
path
[
0
]
else
:
else
:
# buildout injects `sys.path[0:0] = eggs` into python scripts.
# buildout injects `sys.path[0:0] = eggs` into python scripts.
# detect that and remove sys.path entry corresponding to exedir.
# detect that and remove sys.path entry corresponding to exedir.
if
not
_is_buildout_script
(
exe
):
if
not
_is_buildout_script
(
exe
):
raise
RuntimeError
(
'pymain: internal error: sys.path[0] was not set by underlying python to dirname(exe):'
raise
RuntimeError
(
'pymain: internal error: sys.path[0] was not set by underlying python to dirname(exe)
or exe
:'
'
\
n
\
n
\
t
exe:
\
t
%s
\
n
\
t
sys.path[0]:
\
t
%s'
%
(
exe
,
sys
.
path
[
0
]))
'
\
n
\
n
\
t
exe:
\
t
%s
\
n
\
t
sys.path[0]:
\
t
%s'
%
(
exe
,
sys
.
path
[
0
]))
else
:
else
:
if
exedir
in
sys
.
path
:
ok
=
False
sys
.
path
.
remove
(
exedir
)
for
_
in
(
exedir
,
exe
):
else
:
if
_
in
sys
.
path
:
raise
RuntimeError
(
'pymain: internal error: sys.path does not contain dirname(exe):'
sys
.
path
.
remove
(
_
)
ok
=
True
if
not
ok
:
raise
RuntimeError
(
'pymain: internal error: sys.path does not contain dirname(exe) or exe:'
'
\
n
\
n
\
t
exe:
\
t
%s
\
n
\
t
sys.path:
\
t
%s'
%
(
exe
,
sys
.
path
))
'
\
n
\
n
\
t
exe:
\
t
%s
\
n
\
t
sys.path:
\
t
%s'
%
(
exe
,
sys
.
path
))
...
@@ -202,7 +216,6 @@ def pymain(argv, init=None):
...
@@ -202,7 +216,6 @@ def pymain(argv, init=None):
#
#
# python -O gpython file.py
# python -O gpython file.py
if
len
(
reexec_with
)
>
0
:
if
len
(
reexec_with
)
>
0
:
import
os
argv
=
[
sys
.
_gpy_underlying_executable
]
+
reexec_with
+
[
sys
.
executable
]
+
reexec_argv
argv
=
[
sys
.
_gpy_underlying_executable
]
+
reexec_with
+
[
sys
.
executable
]
+
reexec_argv
os
.
execv
(
argv
[
0
],
argv
)
os
.
execv
(
argv
[
0
],
argv
)
...
@@ -439,13 +452,13 @@ def main():
...
@@ -439,13 +452,13 @@ def main():
# _is_buildout_script returns whether file @path is generated as python buildout script.
# _is_buildout_script returns whether file @path is generated as python buildout script.
def
_is_buildout_script
(
path
):
def
_is_buildout_script
(
path
):
with
open
(
path
,
'r'
)
as
f
:
with
open
(
path
,
'r
b
'
)
as
f
:
src
=
f
.
read
()
src
=
f
.
read
()
# buildout injects the following prologues into python scripts:
# buildout injects the following prologues into python scripts:
# sys.path[0:0] = [
# sys.path[0:0] = [
# ...
# ...
# ]
# ]
return
(
'
\
n
sys.path[0:0] = [
\
n
'
in
src
)
return
(
b
'
\
n
sys.path[0:0] = [
\
n
'
in
src
)
# _IGetOpt provides getopt-style incremental options parsing.
# _IGetOpt provides getopt-style incremental options parsing.
...
...
gpython/gpython_test.py
View file @
2b56ce71
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
# Copyright (C) 2019-202
2
Nexedi SA and Contributors.
# Copyright (C) 2019-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -184,7 +184,7 @@ def test_pymain():
...
@@ -184,7 +184,7 @@ def test_pymain():
_
=
pyout
([
'-m'
,
'hello'
,
'abc'
,
'def'
],
cwd
=
testdata
)
_
=
pyout
([
'-m'
,
'hello'
,
'abc'
,
'def'
],
cwd
=
testdata
)
# realpath rewrites e.g. `local/lib -> lib` if local/lib is symlink
# realpath rewrites e.g. `local/lib -> lib` if local/lib is symlink
hellopy
=
realpath
(
join
(
testdata
,
'hello.py'
))
hellopy
=
realpath
(
join
(
testdata
,
'hello.py'
))
assert
_
==
b"hello
\
n
world
\
n
[
'%s', 'abc', 'def']
\
n
"
%
b
(
hellopy
)
assert
_
==
b"hello
\
n
world
\
n
[
%s, 'abc', 'def']
\
n
"
%
b
(
repr
(
hellopy
)
)
# -m<module>
# -m<module>
__
=
pyout
([
'-mhello'
,
'abc'
,
'def'
],
cwd
=
testdata
)
__
=
pyout
([
'-mhello'
,
'abc'
,
'def'
],
cwd
=
testdata
)
assert
__
==
_
assert
__
==
_
...
@@ -195,8 +195,8 @@ def test_pymain():
...
@@ -195,8 +195,8 @@ def test_pymain():
# -i after stdin (also tests interactive mode as -i forces interactive even on non-tty)
# -i after stdin (also tests interactive mode as -i forces interactive even on non-tty)
d
=
{
d
=
{
b'
hellopy'
:
b
(
hellopy
),
b'
repr(hellopy)'
:
b
(
repr
(
hellopy
)
),
b'ps1'
:
b''
# cpython emits prompt to stderr
b'ps1'
:
b''
# cpython emits prompt to stderr
}
}
if
is_pypy
and
not
is_gpython
:
if
is_pypy
and
not
is_gpython
:
d
[
b'ps1'
]
=
b'>>>> '
# native pypy emits prompt to stdout and >>>> instead of >>>
d
[
b'ps1'
]
=
b'>>>> '
# native pypy emits prompt to stdout and >>>> instead of >>>
...
@@ -212,7 +212,7 @@ def test_pymain():
...
@@ -212,7 +212,7 @@ def test_pymain():
assert
_
==
b"hello
\
n
world
\
n
['-c']
\
n
%(ps1)s'~~HELLO~~'
\
n
%(ps1)s"
%
d
assert
_
==
b"hello
\
n
world
\
n
['-c']
\
n
%(ps1)s'~~HELLO~~'
\
n
%(ps1)s"
%
d
# -i after -m
# -i after -m
_
=
pyout
([
'-i'
,
'-m'
,
'hello'
],
stdin
=
b'world.tag'
,
cwd
=
testdata
)
_
=
pyout
([
'-i'
,
'-m'
,
'hello'
],
stdin
=
b'world.tag'
,
cwd
=
testdata
)
assert
_
==
b"hello
\
n
world
\
n
[
'%(hellopy)s'
]
\
n
%(ps1)s'~~WORLD~~'
\
n
%(ps1)s"
%
d
assert
_
==
b"hello
\
n
world
\
n
[
%(repr(hellopy))s
]
\
n
%(ps1)s'~~WORLD~~'
\
n
%(ps1)s"
%
d
# -i after file
# -i after file
_
=
pyout
([
'-i'
,
'testdata/hello.py'
],
stdin
=
b'tag'
,
cwd
=
here
)
_
=
pyout
([
'-i'
,
'testdata/hello.py'
],
stdin
=
b'tag'
,
cwd
=
here
)
assert
_
==
b"hello
\
n
world
\
n
['testdata/hello.py']
\
n
%(ps1)s'~~HELLO~~'
\
n
%(ps1)s"
%
d
assert
_
==
b"hello
\
n
world
\
n
['testdata/hello.py']
\
n
%(ps1)s'~~HELLO~~'
\
n
%(ps1)s"
%
d
...
...
gpython/testprog/print_opt.py
View file @
2b56ce71
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
# Copyright (C) 2020 Nexedi SA and Contributors.
# Copyright (C) 2020
-2023
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
#
Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
# it under the terms of the GNU General Public License version 3, or (at your
# it under the terms of the GNU General Public License version 3, or (at your
...
@@ -61,9 +61,10 @@ def modpy_imports_from():
...
@@ -61,9 +61,10 @@ def modpy_imports_from():
else
:
else
:
raise
AssertionError
(
"module 'mod' is already there"
)
raise
AssertionError
(
"module 'mod' is already there"
)
tmpd
=
tempfile
.
mkdtemp
(
''
,
'modpy_imports_from'
)
tmpd
=
tempfile
.
mkdtemp
(
''
,
'modpy_imports_from'
)
tmpd_
=
tmpd
+
os
.
path
.
sep
try
:
try
:
pymod
=
"%s/mod.py"
%
tmpd
pymod
=
tmpd_
+
"mod.py"
with
open
(
pymod
,
"w"
)
as
f
:
with
open
(
pymod
,
"w"
)
as
f
:
f
.
write
(
"# hello up there
\
n
"
)
f
.
write
(
"# hello up there
\
n
"
)
...
@@ -73,9 +74,9 @@ def modpy_imports_from():
...
@@ -73,9 +74,9 @@ def modpy_imports_from():
files
=
set
()
files
=
set
()
for
dirpath
,
dirnames
,
filenames
in
os
.
walk
(
tmpd
):
for
dirpath
,
dirnames
,
filenames
in
os
.
walk
(
tmpd
):
for
_
in
filenames
:
for
_
in
filenames
:
f
=
'%s/%s'
%
(
dirpath
,
_
)
f
=
os
.
path
.
join
(
dirpath
,
_
)
if
f
.
startswith
(
tmpd
+
'/'
):
if
f
.
startswith
(
tmpd
_
):
f
=
f
[
len
(
tmpd
+
'/'
):]
f
=
f
[
len
(
tmpd
_
):]
files
.
add
(
f
)
files
.
add
(
f
)
...
...
pyproject.toml
View file @
2b56ce71
[build-system]
[build-system]
requires
=
[
"setuptools"
,
"wheel"
,
"setuptools_dso >=
1
.7"
,
"cython"
,
"gevent"
]
requires
=
[
"setuptools"
,
"wheel"
,
"setuptools_dso >=
2
.7"
,
"cython"
,
"gevent"
]
setup.py
View file @
2b56ce71
# pygolang | pythonic package setup
# pygolang | pythonic package setup
# Copyright (C) 2018-202
2
Nexedi SA and Contributors.
# Copyright (C) 2018-202
3
Nexedi SA and Contributors.
# Kirill Smelkov <kirr@nexedi.com>
# Kirill Smelkov <kirr@nexedi.com>
#
#
# This program is free software: you can Use, Study, Modify and Redistribute
# This program is free software: you can Use, Study, Modify and Redistribute
...
@@ -19,28 +19,27 @@
...
@@ -19,28 +19,27 @@
# See https://www.nexedi.com/licensing for rationale and options.
# See https://www.nexedi.com/licensing for rationale and options.
from
setuptools
import
find_packages
from
setuptools
import
find_packages
# setuptools has Library but this days it is not well supported and test for it
# has been killed https://github.com/pypa/setuptools/commit/654c26f78a30
# -> use setuptools_dso instead.
from
setuptools_dso
import
DSO
from
setuptools.command.install_scripts
import
install_scripts
as
_install_scripts
from
setuptools.command.install_scripts
import
install_scripts
as
_install_scripts
from
setuptools.command.develop
import
develop
as
_develop
from
setuptools.command.develop
import
develop
as
_develop
from
distutils
import
sysconfig
from
distutils
import
sysconfig
from
os.path
import
dirname
,
join
from
os.path
import
dirname
,
join
import
sys
,
re
import
sys
,
os
,
re
# read file content
# read file content
def
readfile
(
path
):
def
readfile
(
path
):
# -> str
with
open
(
path
,
'r'
)
as
f
:
with
open
(
path
,
'rb'
)
as
f
:
return
f
.
read
()
data
=
f
.
read
()
if
not
isinstance
(
data
,
str
):
# py3
# reuse golang.pyx.build to build pygolang extensions.
data
=
data
.
decode
(
'utf-8'
)
return
data
# reuse golang.pyx.build to build pygolang dso and extensions.
# we have to be careful and inject synthetic golang package in order to be
# we have to be careful and inject synthetic golang package in order to be
# able to import golang.pyx.build without built/working golang.
# able to import golang.pyx.build without built/working golang.
trun
=
{}
trun
=
{}
exec
(
readfile
(
'trun'
),
trun
)
exec
(
readfile
(
'trun'
),
trun
)
trun
[
'ximport_empty_golangmod'
]()
trun
[
'ximport_empty_golangmod'
]()
from
golang.pyx.build
import
setup
,
Extension
as
Ext
from
golang.pyx.build
import
setup
,
DSO
,
Extension
as
Ext
# grep searches text for pattern.
# grep searches text for pattern.
...
@@ -156,7 +155,7 @@ class develop(XInstallGPython, _develop):
...
@@ -156,7 +155,7 @@ class develop(XInstallGPython, _develop):
# requirements of packages under "golang." namespace
# requirements of packages under "golang." namespace
R
=
{
R
=
{
'cmd.pybench'
:
{
'pytest'
},
'cmd.pybench'
:
{
'pytest'
},
'pyx.build'
:
{
'setuptools'
,
'wheel'
,
'cython'
,
'setuptools_dso >=
1
.7'
},
'pyx.build'
:
{
'setuptools'
,
'wheel'
,
'cython'
,
'setuptools_dso >=
2
.7'
},
'x.perf.benchlib'
:
{
'numpy'
},
'x.perf.benchlib'
:
{
'numpy'
},
}
}
# TODO generate `a.b -> a`, e.g. x.perf = join(x.perf.*); x = join(x.*)
# TODO generate `a.b -> a`, e.g. x.perf = join(x.perf.*); x = join(x.*)
...
@@ -174,6 +173,14 @@ for k in sorted(R.keys()):
...
@@ -174,6 +173,14 @@ for k in sorted(R.keys()):
extras_require
[
k
]
=
list
(
sorted
(
R
[
k
]))
extras_require
[
k
]
=
list
(
sorted
(
R
[
k
]))
# get_python_libdir() returns path where libpython is located
def
get_python_libdir
():
# mimic what distutils.command.build_ext does
if
os
.
name
==
'nt'
:
return
join
(
sysconfig
.
get_config_var
(
'installed_platbase'
),
'libs'
)
else
:
return
sysconfig
.
get_config_var
(
'LIBDIR'
)
setup
(
setup
(
name
=
'pygolang'
,
name
=
'pygolang'
,
version
=
version
,
version
=
version
,
...
@@ -223,19 +230,17 @@ setup(
...
@@ -223,19 +230,17 @@ setup(
'golang/strings.h'
,
'golang/strings.h'
,
'golang/sync.h'
,
'golang/sync.h'
,
'golang/time.h'
],
'golang/time.h'
],
include_dirs
=
[
'
.'
,
'
3rdparty/include'
],
include_dirs
=
[
'3rdparty/include'
],
define_macros
=
[(
'BUILDING_LIBGOLANG'
,
None
)],
define_macros
=
[(
'BUILDING_LIBGOLANG'
,
None
)],
extra_compile_args
=
[
'-std=gnu++11'
],
# not c++11 as linux/list.h uses typeof
soversion
=
'0.1'
),
soversion
=
'0.1'
),
DSO
(
'golang.runtime.libpyxruntime'
,
DSO
(
'golang.runtime.libpyxruntime'
,
[
'golang/runtime/libpyxruntime.cpp'
],
[
'golang/runtime/libpyxruntime.cpp'
],
depends
=
[
'golang/pyx/runtime.h'
],
depends
=
[
'golang/pyx/runtime.h'
],
include_dirs
=
[
'.'
,
sysconfig
.
get_python_inc
()],
include_dirs
=
[
sysconfig
.
get_python_inc
()],
library_dirs
=
[
get_python_libdir
()],
define_macros
=
[(
'BUILDING_LIBPYXRUNTIME'
,
None
)],
define_macros
=
[(
'BUILDING_LIBPYXRUNTIME'
,
None
)],
extra_compile_args
=
[
'-std=c++11'
],
soversion
=
'0.1'
)],
soversion
=
'0.1'
,
dsos
=
[
'golang.runtime.libgolang'
])],
ext_modules
=
[
ext_modules
=
[
Ext
(
'golang._golang'
,
Ext
(
'golang._golang'
,
...
@@ -311,11 +316,12 @@ setup(
...
@@ -311,11 +316,12 @@ setup(
include_package_data
=
True
,
include_package_data
=
True
,
install_requires
=
[
'gevent'
,
'six'
,
'decorator'
,
'Importing;python_version<="2.7"'
,
install_requires
=
[
'gevent'
,
'six'
,
'decorator'
,
'Importing;python_version<="2.7"'
,
# only runtime part: for dylink_prepare_dso
'setuptools_dso >= 2.7'
,
# pyx.build -> setuptools_dso uses multiprocessing
# pyx.build -> setuptools_dso uses multiprocessing
# FIXME geventmp fails on python2, but setuptools_dso
# setuptools_dso uses multiprocessing only on Python3, and only on systems where
# uses multiprocessing only on Python3, so for now we
# mp.get_start_method()!='fork', while geventmp does not work on windows.
# are ok. https://github.com/karellen/geventmp/pull/2
'geventmp ; python_version>="3" and platform_system != "Windows" '
,
'geventmp;python_version>="3"'
,
],
],
extras_require
=
extras_require
,
extras_require
=
extras_require
,
...
@@ -345,8 +351,15 @@ setup(
...
@@ -345,8 +351,15 @@ setup(
Programming Language :: Python :: 3.7
Programming Language :: Python :: 3.7
Programming Language :: Python :: 3.8
Programming Language :: Python :: 3.8
Programming Language :: Python :: 3.9
Programming Language :: Python :: 3.9
Programming Language :: Python :: 3.10
Programming Language :: Python :: 3.11
Programming Language :: Python :: Implementation :: CPython
Programming Language :: Python :: Implementation :: CPython
Programming Language :: Python :: Implementation :: PyPy
Programming Language :: Python :: Implementation :: PyPy
Operating System :: POSIX
Operating System :: POSIX :: Linux
Operating System :: Unix
Operating System :: MacOS
Operating System :: Microsoft :: Windows
Topic :: Software Development :: Interpreters
Topic :: Software Development :: Interpreters
Topic :: Software Development :: Libraries :: Python Modules
\
Topic :: Software Development :: Libraries :: Python Modules
\
"""
.
splitlines
()]
"""
.
splitlines
()]
...
...
tox.ini
View file @
2b56ce71
[tox]
[tox]
envlist
=
envlist
=
{py27d,py27,py37,py38
d,py38,py39d,py39
,pypy,pypy3}-{thread,gevent}
{py27d,py27,py37,py38
,py39d,py39,py310d,py310,py311d,py311
,pypy,pypy3}-{thread,gevent}
# ThreadSanitizer
# ThreadSanitizer
...
@@ -10,28 +10,31 @@ envlist =
...
@@ -10,28 +10,31 @@ envlist =
# (*) PyPy locks its GIL (see RPyGilAcquire) by manually doing atomic cmpxchg
# (*) PyPy locks its GIL (see RPyGilAcquire) by manually doing atomic cmpxchg
# and other games, which TSAN cannot see if PyPy itself was not compiled with
# and other games, which TSAN cannot see if PyPy itself was not compiled with
# -fsanitize=thread.
# -fsanitize=thread.
{py27d,py27,py37,py38
d,py38,py39d,py39
}-{thread
}-tsan
{py27d,py27,py37,py38
,py39d,py39,py310d,py310,py311d,py311
}-{thread
}-tsan
# XXX py*-gevent-tsan would be nice to have, but at present TSAN is not
# XXX py*-gevent-tsan would be nice to have, but at present TSAN is not
# effective with gevent, because it does not understand greenlet "thread"
# effective with gevent, because it does not understand greenlet "thread"
# switching and so perceives the program as having only one thread where races
# switching and so perceives the program as having only one thread where races
# are impossible. Disabled to save time.
# are impossible. Disabled to save time.
# {py27d,py27,py37,py38
d,py38,py39d,py39
}-{ gevent}-tsan
# {py27d,py27,py37,py38
,py39d,py39,py310d,py310,py311d,py311
}-{ gevent}-tsan
# AddressSanitizer
# AddressSanitizer
# XXX asan does not work with gevent: https://github.com/python-greenlet/greenlet/issues/113
# XXX asan does not work with gevent: https://github.com/python-greenlet/greenlet/issues/113
{py27d,py27,py37,py38
d,py38,py39d,py39
,pypy,pypy3}-{thread
}-asan
{py27d,py27,py37,py38
,py39d,py39,py310d,py310,py311d,py311
,pypy,pypy3}-{thread
}-asan
[testenv]
[testenv]
basepython
=
basepython
=
py27d:
python2.7-dbg
py27d:
python2.7-dbg
py27:
python2.7
py27:
python2.7
py37:
python3.7
py37:
python3.7
py38d:
python3.8-dbg
py38:
python3.8
py38:
python3.8
py39d:
python3.9-dbg
py39d:
python3.9-dbg
py39:
python3.9
py39:
python3.9
py310d:
python3.10-dbg
py310:
python3.10
py311d:
python3.11-dbg
py311:
python3.11
pypy:
pypy
pypy:
pypy
pypy3:
pypy3
pypy3:
pypy3
...
@@ -69,5 +72,5 @@ commands=
...
@@ -69,5 +72,5 @@ commands=
# asan/tsan: tell pytest not to capture output - else it is not possible to see
# asan/tsan: tell pytest not to capture output - else it is not possible to see
# reports from sanitizers because they crash tested process on error.
# reports from sanitizers because they crash tested process on error.
# likewise for python debug builds.
# likewise for python debug builds.
asan,tsan,py{27,3
7
}d:
-s
\
asan,tsan,py{27,3
9,310,311
}d:
-s
\
gpython/
golang/
gpython/
golang/
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment