mirror of
https://github.com/janet-lang/janet
synced 2025-11-06 10:33:03 +00:00
Compare commits
167 Commits
installer-
...
v1.39.0
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4e02f27eb9 | ||
|
|
fd234461d7 | ||
|
|
eabb215391 | ||
|
|
deede6bae0 | ||
|
|
697fdcff6d | ||
|
|
ad8a5cb6c7 | ||
|
|
99abada2c2 | ||
|
|
0624936711 | ||
|
|
f764788b36 | ||
|
|
4701bc6543 | ||
|
|
156fb0c999 | ||
|
|
bf34340737 | ||
|
|
20535e8626 | ||
|
|
1ead670e33 | ||
|
|
3ad86108f2 | ||
|
|
0aee7765cf | ||
|
|
4894a4673a | ||
|
|
f00d3199c3 | ||
|
|
e34a8545e6 | ||
|
|
f974c0667b | ||
|
|
ddc122958b | ||
|
|
2e363bf29c | ||
|
|
312f9faae8 | ||
|
|
8c9cd63cb1 | ||
|
|
2af3f21d69 | ||
|
|
c4e3fa03fa | ||
|
|
91b7bcad3d | ||
|
|
8d2a9c1148 | ||
|
|
f1d47bd05a | ||
|
|
58b1491592 | ||
|
|
21a6ed3bd3 | ||
|
|
e815c91e85 | ||
|
|
d96e584869 | ||
|
|
f4ecb5a90f | ||
|
|
f181948aa9 | ||
|
|
bbe6b90331 | ||
|
|
27f01e2664 | ||
|
|
877967966a | ||
|
|
56c5a0ca09 | ||
|
|
f3ad13c2d4 | ||
|
|
8ac4eec370 | ||
|
|
92e91259c3 | ||
|
|
e355cb07e0 | ||
|
|
5bbfcdacd5 | ||
|
|
790a4f2636 | ||
|
|
84bb84b0b7 | ||
|
|
29f2b5c345 | ||
|
|
4643c8fa35 | ||
|
|
a8e2c8e5b8 | ||
|
|
3d3e880f52 | ||
|
|
bef8ba5e06 | ||
|
|
523639362c | ||
|
|
4b6d5e5671 | ||
|
|
a695454dae | ||
|
|
f2eaa5dee8 | ||
|
|
b27c830d90 | ||
|
|
92a852f2df | ||
|
|
647e218bed | ||
|
|
5ebe945ffd | ||
|
|
6254fffad0 | ||
|
|
5705b2f6c7 | ||
|
|
90a33bc88a | ||
|
|
1ba077c87d | ||
|
|
34629ae314 | ||
|
|
3edc4f35b2 | ||
|
|
06d01c099f | ||
|
|
d493eaf485 | ||
|
|
332f123abe | ||
|
|
38e841fc5c | ||
|
|
e8187fdee5 | ||
|
|
2fedb67cb3 | ||
|
|
bdab93c999 | ||
|
|
a9ff8b388f | ||
|
|
b12dfd784e | ||
|
|
e2cc8f2965 | ||
|
|
17524d2ed3 | ||
|
|
d2ee4aa074 | ||
|
|
363e32d455 | ||
|
|
31920e574d | ||
|
|
cf714ed591 | ||
|
|
b458404b41 | ||
|
|
707463a645 | ||
|
|
eac37ab869 | ||
|
|
a24e5b1eaa | ||
|
|
09ac85b1b9 | ||
|
|
87c1eab7d4 | ||
|
|
5a29a28c11 | ||
|
|
2ed186664f | ||
|
|
73334f3485 | ||
|
|
a5b8da8d67 | ||
|
|
e8cccfced5 | ||
|
|
88984f7ffb | ||
|
|
182170b3be | ||
|
|
f92412841b | ||
|
|
18c00e89da | ||
|
|
7c38a55a9a | ||
|
|
a15916ec9c | ||
|
|
3583d4c92f | ||
|
|
a456c67a7b | ||
|
|
6e226e4073 | ||
|
|
d23e6614b1 | ||
|
|
ab6afa72fd | ||
|
|
9538b8a77c | ||
|
|
36d3804dd2 | ||
|
|
a34b8ea68f | ||
|
|
55c10f98bb | ||
|
|
4b5a2a14c0 | ||
|
|
665705d06d | ||
|
|
cc8cd4bace | ||
|
|
8f8b6ed001 | ||
|
|
aa9efee868 | ||
|
|
e0a0e2ed42 | ||
|
|
39f5c539d7 | ||
|
|
1b6437a4f8 | ||
|
|
c62c1a58f0 | ||
|
|
3441bcbd69 | ||
|
|
2e6001316a | ||
|
|
53bcc15207 | ||
|
|
dd609bb1bb | ||
|
|
5c56c7fa91 | ||
|
|
9a892363a3 | ||
|
|
5f550ea5d4 | ||
|
|
1b6cc023a5 | ||
|
|
410f8d69bc | ||
|
|
6da44bdb6a | ||
|
|
d30fd27575 | ||
|
|
1b278fc657 | ||
|
|
eecffe01a5 | ||
|
|
f63a33884f | ||
|
|
fa75a395cb | ||
|
|
1f34ec9902 | ||
|
|
f75c08a78e | ||
|
|
5e93f0e34b | ||
|
|
49f151e265 | ||
|
|
2b73a15ad8 | ||
|
|
06d581dde3 | ||
|
|
2b49903c82 | ||
|
|
a17ae977a5 | ||
|
|
8a6b44cb4e | ||
|
|
60d9f97750 | ||
|
|
f252933f62 | ||
|
|
6dbd7b476c | ||
|
|
a47eb847fb | ||
|
|
ba5990ef21 | ||
|
|
753911fe2d | ||
|
|
746ced5501 | ||
|
|
1b49934e4f | ||
|
|
682f0f584f | ||
|
|
611b2a6c3a | ||
|
|
8043caf581 | ||
|
|
b2d2690eb9 | ||
|
|
7f745a34c3 | ||
|
|
b16cf17246 | ||
|
|
67e8518ba6 | ||
|
|
e94e8dc484 | ||
|
|
1a24d4fc86 | ||
|
|
6ee05785d1 | ||
|
|
268ff666d2 | ||
|
|
91bb34c3bf | ||
|
|
17d5fb3210 | ||
|
|
687b987f7e | ||
|
|
4daecc9a41 | ||
|
|
a85eacadda | ||
|
|
ed63987fd1 | ||
|
|
ff173047f4 | ||
|
|
83e8aab289 | ||
|
|
952906279c |
@@ -1,4 +1,4 @@
|
|||||||
image: openbsd/7.4
|
image: openbsd/7.6
|
||||||
sources:
|
sources:
|
||||||
- https://git.sr.ht/~bakpakin/janet
|
- https://git.sr.ht/~bakpakin/janet
|
||||||
packages:
|
packages:
|
||||||
|
|||||||
7
.github/workflows/codeql.yml
vendored
7
.github/workflows/codeql.yml
vendored
@@ -27,15 +27,16 @@ jobs:
|
|||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v3
|
||||||
|
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@v2
|
uses: github/codeql-action/init@v3
|
||||||
with:
|
with:
|
||||||
languages: ${{ matrix.language }}
|
languages: ${{ matrix.language }}
|
||||||
queries: +security-and-quality
|
queries: +security-and-quality
|
||||||
|
tools: linked
|
||||||
|
|
||||||
- name: Autobuild
|
- name: Autobuild
|
||||||
uses: github/codeql-action/autobuild@v2
|
uses: github/codeql-action/autobuild@v3
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@v2
|
uses: github/codeql-action/analyze@v3
|
||||||
with:
|
with:
|
||||||
category: "/language:${{ matrix.language }}"
|
category: "/language:${{ matrix.language }}"
|
||||||
|
|||||||
12
.github/workflows/test.yml
vendored
12
.github/workflows/test.yml
vendored
@@ -25,7 +25,7 @@ jobs:
|
|||||||
name: Build and test on Windows
|
name: Build and test on Windows
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
os: [ windows-latest, windows-2019 ]
|
os: [ windows-latest, windows-2022 ]
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
@@ -46,7 +46,7 @@ jobs:
|
|||||||
name: Build and test on Windows Minimal build
|
name: Build and test on Windows Minimal build
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
os: [ windows-2019 ]
|
os: [ windows-2022 ]
|
||||||
runs-on: ${{ matrix.os }}
|
runs-on: ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
@@ -132,7 +132,7 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@master
|
uses: actions/checkout@master
|
||||||
- name: Do Qemu build and test
|
- name: Enable qemu
|
||||||
run: |
|
run: docker run --privileged --rm tonistiigi/binfmt --install s390x
|
||||||
docker run --rm --privileged multiarch/qemu-user-static --reset -p yes
|
- name: Build and run on emulated architecture
|
||||||
docker run --rm -v .:/janet --platform linux/s390x ubuntu bash -c "apt-get -y update && apt-get -y install git build-essential && cd /janet && make -j3 && make test"
|
run: docker run --rm -v .:/janet --platform linux/s390x alpine sh -c "apk update && apk add --no-interactive git build-base && cd /janet && make -j3 && make test"
|
||||||
|
|||||||
28
CHANGELOG.md
28
CHANGELOG.md
@@ -1,7 +1,33 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
All notable changes to this project will be documented in this file.
|
All notable changes to this project will be documented in this file.
|
||||||
|
|
||||||
## 1.37.0 - 2024-12-05
|
## 1.39.0 - 2025-08-24
|
||||||
|
- Various bug fixes
|
||||||
|
- Add `net/socket`
|
||||||
|
- Add support for illumos OS
|
||||||
|
- Raise helpful errors for incorrect arguments to `import`.
|
||||||
|
- Allow configuring `JANET_THREAD_LOCAL` during builds to allow multi-threading on unknown compilers.
|
||||||
|
- Make `ffi/write` append to a buffer instead of insert at 0 by default.
|
||||||
|
- Add `os/getpid` to get the current process id.
|
||||||
|
- Add `:out` option to `os/spawn` to be able to redirect stderr to stdout with pipes.
|
||||||
|
Add `interrupt?` argument to `ev/deadline` to use VM interruptions.
|
||||||
|
|
||||||
|
## 1.38.0 - 2025-03-18
|
||||||
|
- Add `bundle/replace`
|
||||||
|
- Add CLI flags for the `bundle/` module to install and manage bundles.
|
||||||
|
- Improve `?` peg special termination behavior
|
||||||
|
- Add IEEE hex floats to grammar.
|
||||||
|
- Add buffer peg literal support
|
||||||
|
- Improve `split` peg special edge case behavior
|
||||||
|
- Add Arm64 .msi support
|
||||||
|
- Add `no-reuse` argument to `net/listen` to disable reusing server sockets
|
||||||
|
- Add `struct/rawget`
|
||||||
|
- Fix `deep=` and `deep-not=` to better handle degenerate cases with mutable table keys
|
||||||
|
- Long strings will now dedent on `\r\n` instead of just `\n`.
|
||||||
|
- Add `ev/to-file` for synchronous resource operations
|
||||||
|
- Improve `file/open` error message by including path
|
||||||
|
|
||||||
|
## 1.37.1 - 2024-12-05
|
||||||
- Fix meson cross compilation
|
- Fix meson cross compilation
|
||||||
- Update timeout documentation for networking APIs: timeouts raise errors and do not return nil.
|
- Update timeout documentation for networking APIs: timeouts raise errors and do not return nil.
|
||||||
- Add `janet_addtimeout_nil(double sec);` to the C API.
|
- Add `janet_addtimeout_nil(double sec);` to the C API.
|
||||||
|
|||||||
2
LICENSE
2
LICENSE
@@ -1,4 +1,4 @@
|
|||||||
Copyright (c) 2023 Calvin Rose and contributors
|
Copyright (c) 2025 Calvin Rose and contributors
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
this software and associated documentation files (the "Software"), to deal in
|
this software and associated documentation files (the "Software"), to deal in
|
||||||
|
|||||||
31
Makefile
31
Makefile
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2024 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
@@ -47,6 +47,7 @@ SPORK_TAG?=master
|
|||||||
HAS_SHARED?=1
|
HAS_SHARED?=1
|
||||||
DEBUGGER=gdb
|
DEBUGGER=gdb
|
||||||
SONAME_SETTER=-Wl,-soname,
|
SONAME_SETTER=-Wl,-soname,
|
||||||
|
STRIPFLAGS=-x -S
|
||||||
|
|
||||||
# For cross compilation
|
# For cross compilation
|
||||||
HOSTCC?=$(CC)
|
HOSTCC?=$(CC)
|
||||||
@@ -54,9 +55,10 @@ HOSTAR?=$(AR)
|
|||||||
# Symbols are (optionally) removed later, keep -g as default!
|
# Symbols are (optionally) removed later, keep -g as default!
|
||||||
CFLAGS?=-O2 -g
|
CFLAGS?=-O2 -g
|
||||||
LDFLAGS?=-rdynamic
|
LDFLAGS?=-rdynamic
|
||||||
LIBJANET_LDFLAGS?=$(LD_FLAGS)
|
LIBJANET_LDFLAGS?=$(LDFLAGS)
|
||||||
RUN:=$(RUN)
|
RUN:=$(RUN)
|
||||||
|
|
||||||
|
|
||||||
COMMON_CFLAGS:=-std=c99 -Wall -Wextra -Isrc/include -Isrc/conf -fvisibility=hidden -fPIC
|
COMMON_CFLAGS:=-std=c99 -Wall -Wextra -Isrc/include -Isrc/conf -fvisibility=hidden -fPIC
|
||||||
BOOT_CFLAGS:=-DJANET_BOOTSTRAP -DJANET_BUILD=$(JANET_BUILD) -O0 $(COMMON_CFLAGS) -g
|
BOOT_CFLAGS:=-DJANET_BOOTSTRAP -DJANET_BUILD=$(JANET_BUILD) -O0 $(COMMON_CFLAGS) -g
|
||||||
BUILD_CFLAGS:=$(CFLAGS) $(COMMON_CFLAGS)
|
BUILD_CFLAGS:=$(CFLAGS) $(COMMON_CFLAGS)
|
||||||
@@ -79,6 +81,12 @@ ifeq ($(UNAME), Darwin)
|
|||||||
LDCONFIG:=true
|
LDCONFIG:=true
|
||||||
else ifeq ($(UNAME), Linux)
|
else ifeq ($(UNAME), Linux)
|
||||||
CLIBS:=$(CLIBS) -lrt -ldl
|
CLIBS:=$(CLIBS) -lrt -ldl
|
||||||
|
else ifeq ($(UNAME), SunOS)
|
||||||
|
BUILD_CFLAGS+=-D__EXTENSIONS__ -DJANET_NO_NANBOX
|
||||||
|
BOOT_CFLAGS+=-D__EXTENSIONS__ -DJANET_NO_NANBOX
|
||||||
|
CLIBS:=-lsocket -lm
|
||||||
|
STRIPFLAGS=-x
|
||||||
|
LDCONFIG:=false
|
||||||
endif
|
endif
|
||||||
|
|
||||||
# For other unix likes, add flags here!
|
# For other unix likes, add flags here!
|
||||||
@@ -94,12 +102,18 @@ endif
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
# Mingw
|
# Mingw
|
||||||
|
MINGW_COMPILER=
|
||||||
ifeq ($(findstring MINGW,$(UNAME)), MINGW)
|
ifeq ($(findstring MINGW,$(UNAME)), MINGW)
|
||||||
|
MINGW_COMPILER=gcc
|
||||||
CLIBS:=-lws2_32 -lpsapi -lwsock32
|
CLIBS:=-lws2_32 -lpsapi -lwsock32
|
||||||
LDFLAGS:=-Wl,--out-implib,$(JANET_IMPORT_LIB)
|
LDFLAGS:=-Wl,--out-implib,$(JANET_IMPORT_LIB)
|
||||||
LIBJANET_LDFLAGS:=-Wl,--out-implib,$(JANET_LIBRARY_IMPORT_LIB)
|
LIBJANET_LDFLAGS:=-Wl,--out-implib,$(JANET_LIBRARY_IMPORT_LIB)
|
||||||
JANET_TARGET:=$(JANET_TARGET).exe
|
JANET_TARGET:=$(JANET_TARGET).exe
|
||||||
JANET_BOOT:=$(JANET_BOOT).exe
|
JANET_BOOT:=$(JANET_BOOT).exe
|
||||||
|
COMPILER_VERSION:=$(shell $(CC) --version)
|
||||||
|
ifeq ($(findstring clang,$(COMPILER_VERSION)), clang)
|
||||||
|
MINGW_COMPILER=clang
|
||||||
|
endif
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
|
||||||
@@ -206,9 +220,14 @@ build/%.bin.o: src/%.c $(JANET_HEADERS) $(JANET_LOCAL_HEADERS) Makefile
|
|||||||
########################
|
########################
|
||||||
|
|
||||||
ifeq ($(UNAME), Darwin)
|
ifeq ($(UNAME), Darwin)
|
||||||
SONAME=libjanet.1.37.dylib
|
SONAME=libjanet.1.39.dylib
|
||||||
else
|
else
|
||||||
SONAME=libjanet.so.1.37
|
SONAME=libjanet.so.1.39
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifeq ($(MINGW_COMPILER), clang)
|
||||||
|
SONAME=
|
||||||
|
SONAME_SETTER=
|
||||||
endif
|
endif
|
||||||
|
|
||||||
build/c/shell.c: src/mainclient/shell.c
|
build/c/shell.c: src/mainclient/shell.c
|
||||||
@@ -277,7 +296,7 @@ build/janet-%.tar.gz: $(JANET_TARGET) \
|
|||||||
README.md build/c/janet.c build/c/shell.c
|
README.md build/c/janet.c build/c/shell.c
|
||||||
mkdir -p build/$(JANET_DIST_DIR)/bin
|
mkdir -p build/$(JANET_DIST_DIR)/bin
|
||||||
cp $(JANET_TARGET) build/$(JANET_DIST_DIR)/bin/
|
cp $(JANET_TARGET) build/$(JANET_DIST_DIR)/bin/
|
||||||
strip -x -S 'build/$(JANET_DIST_DIR)/bin/janet'
|
strip $(STRIPFLAGS) 'build/$(JANET_DIST_DIR)/bin/janet'
|
||||||
mkdir -p build/$(JANET_DIST_DIR)/include
|
mkdir -p build/$(JANET_DIST_DIR)/include
|
||||||
cp build/janet.h build/$(JANET_DIST_DIR)/include/
|
cp build/janet.h build/$(JANET_DIST_DIR)/include/
|
||||||
mkdir -p build/$(JANET_DIST_DIR)/lib/
|
mkdir -p build/$(JANET_DIST_DIR)/lib/
|
||||||
@@ -324,7 +343,7 @@ build/janet.pc: $(JANET_TARGET)
|
|||||||
install: $(JANET_TARGET) $(JANET_LIBRARY) $(JANET_STATIC_LIBRARY) build/janet.pc build/janet.h
|
install: $(JANET_TARGET) $(JANET_LIBRARY) $(JANET_STATIC_LIBRARY) build/janet.pc build/janet.h
|
||||||
mkdir -p '$(DESTDIR)$(BINDIR)'
|
mkdir -p '$(DESTDIR)$(BINDIR)'
|
||||||
cp $(JANET_TARGET) '$(DESTDIR)$(BINDIR)/janet'
|
cp $(JANET_TARGET) '$(DESTDIR)$(BINDIR)/janet'
|
||||||
strip -x -S '$(DESTDIR)$(BINDIR)/janet'
|
strip $(STRIPFLAGS) '$(DESTDIR)$(BINDIR)/janet'
|
||||||
mkdir -p '$(DESTDIR)$(INCLUDEDIR)/janet'
|
mkdir -p '$(DESTDIR)$(INCLUDEDIR)/janet'
|
||||||
cp -r build/janet.h '$(DESTDIR)$(INCLUDEDIR)/janet'
|
cp -r build/janet.h '$(DESTDIR)$(INCLUDEDIR)/janet'
|
||||||
ln -sf ./janet/janet.h '$(DESTDIR)$(INCLUDEDIR)/janet.h'
|
ln -sf ./janet/janet.h '$(DESTDIR)$(INCLUDEDIR)/janet.h'
|
||||||
|
|||||||
21
README.md
21
README.md
@@ -165,6 +165,21 @@ make install-jpm-git
|
|||||||
|
|
||||||
Find out more about the available make targets by running `make help`.
|
Find out more about the available make targets by running `make help`.
|
||||||
|
|
||||||
|
### Alpine Linux
|
||||||
|
|
||||||
|
To build a statically-linked build of Janet, Alpine Linux + MUSL is a good combination. Janet can also
|
||||||
|
be built inside a docker container or similar in this manner.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
docker run -it --rm alpine /bin/ash
|
||||||
|
$ apk add make gcc musl-dev git
|
||||||
|
$ git clone https://github.com/janet-lang/janet.git
|
||||||
|
$ cd janet
|
||||||
|
$ make -j10
|
||||||
|
$ make test
|
||||||
|
$ make install
|
||||||
|
```
|
||||||
|
|
||||||
### 32-bit Haiku
|
### 32-bit Haiku
|
||||||
|
|
||||||
32-bit Haiku build instructions are the same as the UNIX-like build instructions,
|
32-bit Haiku build instructions are the same as the UNIX-like build instructions,
|
||||||
@@ -198,6 +213,10 @@ gmake install-jpm-git
|
|||||||
NetBSD build instructions are the same as the FreeBSD build instructions.
|
NetBSD build instructions are the same as the FreeBSD build instructions.
|
||||||
Alternatively, install the package directly with `pkgin install janet`.
|
Alternatively, install the package directly with `pkgin install janet`.
|
||||||
|
|
||||||
|
### illumos
|
||||||
|
|
||||||
|
Building on illumos is exactly the same as building on FreeBSD.
|
||||||
|
|
||||||
### Windows
|
### Windows
|
||||||
|
|
||||||
1. Install [Visual Studio](https://visualstudio.microsoft.com/thank-you-downloading-visual-studio/?sku=Community&rel=15#) or [Visual Studio Build Tools](https://visualstudio.microsoft.com/thank-you-downloading-visual-studio/?sku=BuildTools&rel=15#).
|
1. Install [Visual Studio](https://visualstudio.microsoft.com/thank-you-downloading-visual-studio/?sku=Community&rel=15#) or [Visual Studio Build Tools](https://visualstudio.microsoft.com/thank-you-downloading-visual-studio/?sku=BuildTools&rel=15#).
|
||||||
@@ -207,7 +226,7 @@ Alternatively, install the package directly with `pkgin install janet`.
|
|||||||
|
|
||||||
To build an `.msi` installer executable, in addition to the above steps, you will have to:
|
To build an `.msi` installer executable, in addition to the above steps, you will have to:
|
||||||
|
|
||||||
5. Install, or otherwise add to your PATH the [WiX 3.11 Toolset](https://github.com/wixtoolset/wix3/releases).
|
5. Install, or otherwise add to your PATH the [WiX 3.14 Toolset](https://github.com/wixtoolset/wix3/releases).
|
||||||
6. Run `build_win dist`.
|
6. Run `build_win dist`.
|
||||||
|
|
||||||
Now you should have an `.msi`. You can run `build_win install` to install the `.msi`, or execute the file itself.
|
Now you should have an `.msi`. You can run `build_win install` to install the `.msi`, or execute the file itself.
|
||||||
|
|||||||
@@ -41,34 +41,34 @@ if not exist build\boot mkdir build\boot
|
|||||||
@rem Build the bootstrap interpreter
|
@rem Build the bootstrap interpreter
|
||||||
for %%f in (src\core\*.c) do (
|
for %%f in (src\core\*.c) do (
|
||||||
%JANET_COMPILE% /DJANET_BOOTSTRAP /Fobuild\boot\%%~nf.obj %%f
|
%JANET_COMPILE% /DJANET_BOOTSTRAP /Fobuild\boot\%%~nf.obj %%f
|
||||||
@if not errorlevel 0 goto :BUILDFAIL
|
@if errorlevel 1 goto :BUILDFAIL
|
||||||
)
|
)
|
||||||
for %%f in (src\boot\*.c) do (
|
for %%f in (src\boot\*.c) do (
|
||||||
%JANET_COMPILE% /DJANET_BOOTSTRAP /Fobuild\boot\%%~nf.obj %%f
|
%JANET_COMPILE% /DJANET_BOOTSTRAP /Fobuild\boot\%%~nf.obj %%f
|
||||||
@if not errorlevel 0 goto :BUILDFAIL
|
@if errorlevel 1 goto :BUILDFAIL
|
||||||
)
|
)
|
||||||
%JANET_LINK% /out:build\janet_boot.exe build\boot\*.obj
|
%JANET_LINK% /out:build\janet_boot.exe build\boot\*.obj
|
||||||
@if not errorlevel 0 goto :BUILDFAIL
|
@if errorlevel 1 goto :BUILDFAIL
|
||||||
build\janet_boot . > build\c\janet.c
|
build\janet_boot . > build\c\janet.c
|
||||||
@if not errorlevel 0 goto :BUILDFAIL
|
@if errorlevel 1 goto :BUILDFAIL
|
||||||
|
|
||||||
@rem Build the sources
|
@rem Build the sources
|
||||||
%JANET_COMPILE% /Fobuild\janet.obj build\c\janet.c
|
%JANET_COMPILE% /Fobuild\janet.obj build\c\janet.c
|
||||||
@if not errorlevel 0 goto :BUILDFAIL
|
@if errorlevel 1 goto :BUILDFAIL
|
||||||
%JANET_COMPILE% /Fobuild\shell.obj src\mainclient\shell.c
|
%JANET_COMPILE% /Fobuild\shell.obj src\mainclient\shell.c
|
||||||
@if not errorlevel 0 goto :BUILDFAIL
|
@if errorlevel 1 goto :BUILDFAIL
|
||||||
|
|
||||||
@rem Build the resources
|
@rem Build the resources
|
||||||
rc /nologo /fobuild\janet_win.res janet_win.rc
|
rc /nologo /fobuild\janet_win.res janet_win.rc
|
||||||
@if not errorlevel 0 goto :BUILDFAIL
|
@if errorlevel 1 goto :BUILDFAIL
|
||||||
|
|
||||||
@rem Link everything to main client
|
@rem Link everything to main client
|
||||||
%JANET_LINK% /out:janet.exe build\janet.obj build\shell.obj build\janet_win.res
|
%JANET_LINK% /out:janet.exe build\janet.obj build\shell.obj build\janet_win.res
|
||||||
@if not errorlevel 0 goto :BUILDFAIL
|
@if errorlevel 1 goto :BUILDFAIL
|
||||||
|
|
||||||
@rem Build static library (libjanet.lib)
|
@rem Build static library (libjanet.lib)
|
||||||
%JANET_LINK_STATIC% /out:build\libjanet.lib build\janet.obj
|
%JANET_LINK_STATIC% /out:build\libjanet.lib build\janet.obj
|
||||||
@if not errorlevel 0 goto :BUILDFAIL
|
@if errorlevel 1 goto :BUILDFAIL
|
||||||
|
|
||||||
echo === Successfully built janet.exe for Windows ===
|
echo === Successfully built janet.exe for Windows ===
|
||||||
echo === Run 'build_win test' to run tests. ==
|
echo === Run 'build_win test' to run tests. ==
|
||||||
@@ -91,7 +91,7 @@ exit /b 0
|
|||||||
|
|
||||||
@rem Clean build artifacts
|
@rem Clean build artifacts
|
||||||
:CLEAN
|
:CLEAN
|
||||||
del *.exe *.lib *.exp
|
del *.exe *.lib *.exp *.msi *.wixpdb
|
||||||
rd /s /q build
|
rd /s /q build
|
||||||
if exist dist (
|
if exist dist (
|
||||||
rd /s /q dist
|
rd /s /q dist
|
||||||
@@ -102,7 +102,7 @@ exit /b 0
|
|||||||
:TEST
|
:TEST
|
||||||
for %%f in (test/suite*.janet) do (
|
for %%f in (test/suite*.janet) do (
|
||||||
janet.exe test\%%f
|
janet.exe test\%%f
|
||||||
@if not errorlevel 0 goto TESTFAIL
|
@if errorlevel 1 goto TESTFAIL
|
||||||
)
|
)
|
||||||
exit /b 0
|
exit /b 0
|
||||||
|
|
||||||
@@ -143,7 +143,13 @@ if defined CI (
|
|||||||
) else (
|
) else (
|
||||||
set WIXBIN=
|
set WIXBIN=
|
||||||
)
|
)
|
||||||
%WIXBIN%candle.exe tools\msi\janet.wxs -arch %BUILDARCH% -out build\
|
|
||||||
|
set WIXARCH=%BUILDARCH%
|
||||||
|
if "%WIXARCH%"=="aarch64" (
|
||||||
|
set WIXARCH=arm64
|
||||||
|
)
|
||||||
|
|
||||||
|
%WIXBIN%candle.exe tools\msi\janet.wxs -arch %WIXARCH% -out build\
|
||||||
%WIXBIN%light.exe "-sice:ICE38" -b tools\msi -ext WixUIExtension build\janet.wixobj -out janet-%RELEASE_VERSION%-windows-%BUILDARCH%-installer.msi
|
%WIXBIN%light.exe "-sice:ICE38" -b tools\msi -ext WixUIExtension build\janet.wixobj -out janet-%RELEASE_VERSION%-windows-%BUILDARCH%-installer.msi
|
||||||
exit /b 0
|
exit /b 0
|
||||||
|
|
||||||
|
|||||||
6
examples/abstract-unix-socket.janet
Normal file
6
examples/abstract-unix-socket.janet
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
# Linux only - uses abstract unix domain sockets
|
||||||
|
(ev/spawn (net/server :unix "@abc123" (fn [conn] (print (:read conn 1024)) (:close conn))))
|
||||||
|
(ev/sleep 1)
|
||||||
|
(def s (net/connect :unix "@abc123" :stream))
|
||||||
|
(:write s "hello")
|
||||||
|
(:close s)
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
@{
|
@{
|
||||||
:name "sample-dep1"
|
:name "sample-dep1"
|
||||||
:dependencies ["sample-dep2"]
|
:dependencies [{:name "sample-dep2"}]
|
||||||
}
|
}
|
||||||
|
|||||||
20
meson.build
20
meson.build
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2024 Calvin Rose and contributors
|
# Copyright (c) 2025 Calvin Rose and contributors
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
@@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
project('janet', 'c',
|
project('janet', 'c',
|
||||||
default_options : ['c_std=c99', 'build.c_std=c99', 'b_lundef=false', 'default_library=both'],
|
default_options : ['c_std=c99', 'build.c_std=c99', 'b_lundef=false', 'default_library=both'],
|
||||||
version : '1.37.0')
|
version : '1.38.0')
|
||||||
|
|
||||||
# Global settings
|
# Global settings
|
||||||
janet_path = join_paths(get_option('prefix'), get_option('libdir'), 'janet')
|
janet_path = join_paths(get_option('prefix'), get_option('libdir'), 'janet')
|
||||||
@@ -39,6 +39,15 @@ native_thread_dep = dependency('threads', native : true)
|
|||||||
# Deps
|
# Deps
|
||||||
m_dep = cc.find_library('m', required : false)
|
m_dep = cc.find_library('m', required : false)
|
||||||
dl_dep = cc.find_library('dl', required : false)
|
dl_dep = cc.find_library('dl', required : false)
|
||||||
|
|
||||||
|
# for MINGW/MSYS2
|
||||||
|
native_ws2_dep = native_cc.find_library('ws2_32', required: false)
|
||||||
|
native_psapi_dep = native_cc.find_library('psapi', required: false)
|
||||||
|
native_wsock_dep = native_cc.find_library('wsock32', required: false)
|
||||||
|
ws2_dep = cc.find_library('ws2_32', required: false)
|
||||||
|
psapi_dep = cc.find_library('psapi', required: false)
|
||||||
|
wsock_dep = cc.find_library('wsock32', required: false)
|
||||||
|
|
||||||
android_spawn_dep = cc.find_library('android-spawn', required : false)
|
android_spawn_dep = cc.find_library('android-spawn', required : false)
|
||||||
thread_dep = dependency('threads')
|
thread_dep = dependency('threads')
|
||||||
|
|
||||||
@@ -96,6 +105,9 @@ endif
|
|||||||
if get_option('arch_name') != ''
|
if get_option('arch_name') != ''
|
||||||
conf.set('JANET_ARCH_NAME', get_option('arch_name'))
|
conf.set('JANET_ARCH_NAME', get_option('arch_name'))
|
||||||
endif
|
endif
|
||||||
|
if get_option('thread_local_prefix') != ''
|
||||||
|
conf.set('JANET_THREAD_LOCAL', get_option('thread_local_prefix'))
|
||||||
|
endif
|
||||||
jconf = configure_file(output : 'janetconf.h',
|
jconf = configure_file(output : 'janetconf.h',
|
||||||
configuration : conf)
|
configuration : conf)
|
||||||
|
|
||||||
@@ -173,8 +185,8 @@ mainclient_src = [
|
|||||||
'src/mainclient/shell.c'
|
'src/mainclient/shell.c'
|
||||||
]
|
]
|
||||||
|
|
||||||
janet_dependencies = [m_dep, dl_dep, android_spawn_dep]
|
janet_dependencies = [m_dep, dl_dep, android_spawn_dep, ws2_dep, psapi_dep, wsock_dep]
|
||||||
janet_native_dependencies = [native_m_dep, native_dl_dep, native_android_spawn_dep]
|
janet_native_dependencies = [native_m_dep, native_dl_dep, native_android_spawn_dep, native_ws2_dep, native_psapi_dep, native_wsock_dep]
|
||||||
if not get_option('single_threaded')
|
if not get_option('single_threaded')
|
||||||
janet_dependencies += thread_dep
|
janet_dependencies += thread_dep
|
||||||
janet_native_dependencies += native_thread_dep
|
janet_native_dependencies += native_thread_dep
|
||||||
|
|||||||
@@ -30,6 +30,7 @@ option('max_macro_expand', type : 'integer', min : 1, max : 8000, value : 200)
|
|||||||
option('stack_max', type : 'integer', min : 8096, max : 0x7fffffff, value : 0x7fffffff)
|
option('stack_max', type : 'integer', min : 8096, max : 0x7fffffff, value : 0x7fffffff)
|
||||||
|
|
||||||
option('arch_name', type : 'string', value: '')
|
option('arch_name', type : 'string', value: '')
|
||||||
|
option('thread_local_prefix', type : 'string', value: '')
|
||||||
option('os_name', type : 'string', value: '')
|
option('os_name', type : 'string', value: '')
|
||||||
option('shared', type : 'boolean', value: true)
|
option('shared', type : 'boolean', value: true)
|
||||||
option('cryptorand', type : 'boolean', value: true)
|
option('cryptorand', type : 'boolean', value: true)
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
# The core janet library
|
# The core janet library
|
||||||
# Copyright 2024 © Calvin Rose
|
# Copyright 2025 © Calvin Rose
|
||||||
|
|
||||||
###
|
###
|
||||||
###
|
###
|
||||||
@@ -158,7 +158,7 @@
|
|||||||
``Define an alias for a keyword that is used as a dynamic binding. The
|
``Define an alias for a keyword that is used as a dynamic binding. The
|
||||||
alias is a normal, lexically scoped binding that can be used instead of
|
alias is a normal, lexically scoped binding that can be used instead of
|
||||||
a keyword to prevent typos. `defdyn` does not set dynamic bindings or otherwise
|
a keyword to prevent typos. `defdyn` does not set dynamic bindings or otherwise
|
||||||
replace `dyn` and `setdyn`. The alias _must_ start and end with the `*` character, usually
|
replace `dyn` and `setdyn`. The alias *must* start and end with the `*` character, usually
|
||||||
called "earmuffs".``
|
called "earmuffs".``
|
||||||
[alias & more]
|
[alias & more]
|
||||||
(assert (symbol? alias) "alias must be a symbol")
|
(assert (symbol? alias) "alias must be a symbol")
|
||||||
@@ -290,22 +290,6 @@
|
|||||||
(array/concat accum body)
|
(array/concat accum body)
|
||||||
(tuple/slice accum 0))
|
(tuple/slice accum 0))
|
||||||
|
|
||||||
(defmacro try
|
|
||||||
``Try something and catch errors. `body` is any expression,
|
|
||||||
and `catch` should be a form, the first element of which is a tuple. This tuple
|
|
||||||
should contain a binding for errors and an optional binding for
|
|
||||||
the fiber wrapping the body. Returns the result of `body` if no error,
|
|
||||||
or the result of `catch` if an error.``
|
|
||||||
[body catch]
|
|
||||||
(let [[[err fib]] catch
|
|
||||||
f (gensym)
|
|
||||||
r (gensym)]
|
|
||||||
~(let [,f (,fiber/new (fn :try [] ,body) :ie)
|
|
||||||
,r (,resume ,f)]
|
|
||||||
(if (,= (,fiber/status ,f) :error)
|
|
||||||
(do (def ,err ,r) ,(if fib ~(def ,fib ,f)) ,;(tuple/slice catch 1))
|
|
||||||
,r))))
|
|
||||||
|
|
||||||
(defmacro protect
|
(defmacro protect
|
||||||
`Evaluate expressions, while capturing any errors. Evaluates to a tuple
|
`Evaluate expressions, while capturing any errors. Evaluates to a tuple
|
||||||
of two elements. The first element is true if successful, false if an
|
of two elements. The first element is true if successful, false if an
|
||||||
@@ -352,6 +336,23 @@
|
|||||||
(tuple 'if $fi $fi ret))))))
|
(tuple 'if $fi $fi ret))))))
|
||||||
ret)
|
ret)
|
||||||
|
|
||||||
|
(defmacro try
|
||||||
|
``Try something and catch errors. `body` is any expression,
|
||||||
|
and `catch` should be a form, the first element of which is a tuple. This tuple
|
||||||
|
should contain a binding for errors and an optional binding for
|
||||||
|
the fiber wrapping the body. Returns the result of `body` if no error,
|
||||||
|
or the result of `catch` if an error.``
|
||||||
|
[body catch]
|
||||||
|
(assert (and (not (empty? catch)) (indexed? (catch 0))) "the first element of `catch` must be a tuple or array")
|
||||||
|
(let [[err fib] (catch 0)
|
||||||
|
r (or err (gensym))
|
||||||
|
f (or fib (gensym))]
|
||||||
|
~(let [,f (,fiber/new (fn :try [] ,body) :ie)
|
||||||
|
,r (,resume ,f)]
|
||||||
|
(if (,= (,fiber/status ,f) :error)
|
||||||
|
(do ,;(tuple/slice catch 1))
|
||||||
|
,r))))
|
||||||
|
|
||||||
(defmacro with-syms
|
(defmacro with-syms
|
||||||
"Evaluates `body` with each symbol in `syms` bound to a generated, unique symbol."
|
"Evaluates `body` with each symbol in `syms` bound to a generated, unique symbol."
|
||||||
[syms & body]
|
[syms & body]
|
||||||
@@ -996,7 +997,7 @@
|
|||||||
|
|
||||||
(defn reduce2
|
(defn reduce2
|
||||||
``The 2-argument version of `reduce` that does not take an initialization value.
|
``The 2-argument version of `reduce` that does not take an initialization value.
|
||||||
Instead, the first element of the array is used for initialization.``
|
Instead, the first element of the array is used for initialization. If `ind` is empty, will evaluate to nil.``
|
||||||
[f ind]
|
[f ind]
|
||||||
(var k (next ind))
|
(var k (next ind))
|
||||||
(if (= nil k) (break nil))
|
(if (= nil k) (break nil))
|
||||||
@@ -1084,16 +1085,29 @@
|
|||||||
(map-aggregator ,maptype ,res (,f x ;call-buffer)))))))
|
(map-aggregator ,maptype ,res (,f x ;call-buffer)))))))
|
||||||
|
|
||||||
(defn map
|
(defn map
|
||||||
`Map a function over every value in a data structure and
|
```
|
||||||
return an array of the results.`
|
Map a function `f` over every value in a data structure `ind`
|
||||||
|
and return an array of results, but only if no `inds` are
|
||||||
|
provided. Multiple data structures can be handled if each
|
||||||
|
`inds` is a data structure and `f` is a function of arity
|
||||||
|
one more than the number of `inds`. The resulting array has
|
||||||
|
a length that is the shortest of `ind` and each of `inds`.
|
||||||
|
```
|
||||||
[f ind & inds]
|
[f ind & inds]
|
||||||
(def res @[])
|
(def res @[])
|
||||||
(map-template :map res f ind inds)
|
(map-template :map res f ind inds)
|
||||||
res)
|
res)
|
||||||
|
|
||||||
(defn mapcat
|
(defn mapcat
|
||||||
``Map a function over every element in an array or tuple and
|
```
|
||||||
use `array/concat` to concatenate the results.``
|
Map a function `f` over every value in a data structure `ind`
|
||||||
|
and use `array/concat` to concatenate the results, but only if
|
||||||
|
no `inds` are provided. Multiple data structures can be handled
|
||||||
|
if each `inds` is a data structure and `f` is a function of
|
||||||
|
arity one more than the number of `inds`. Note that `f` is only
|
||||||
|
applied to values at indeces up to the largest index of the
|
||||||
|
shortest of `ind` and each of `inds`.
|
||||||
|
```
|
||||||
[f ind & inds]
|
[f ind & inds]
|
||||||
(def res @[])
|
(def res @[])
|
||||||
(map-template :mapcat res f ind inds)
|
(map-template :mapcat res f ind inds)
|
||||||
@@ -1110,18 +1124,30 @@
|
|||||||
res)
|
res)
|
||||||
|
|
||||||
(defn count
|
(defn count
|
||||||
``Count the number of items in `ind` for which `(pred item)`
|
```
|
||||||
is true.``
|
Count the number of values in a data structure `ind` for which
|
||||||
|
applying `pred` yields a truthy value, but only if no `inds` are
|
||||||
|
provided. Multiple data structures can be handled if each `inds`
|
||||||
|
is a data structure and `pred` is a function of arity one more
|
||||||
|
than the number of `inds`. Note that `pred` is only applied to
|
||||||
|
values at indeces up to the largest index of the shortest of
|
||||||
|
`ind` and each of `inds`.
|
||||||
|
```
|
||||||
[pred ind & inds]
|
[pred ind & inds]
|
||||||
(var res 0)
|
(var res 0)
|
||||||
(map-template :count res pred ind inds)
|
(map-template :count res pred ind inds)
|
||||||
res)
|
res)
|
||||||
|
|
||||||
(defn keep
|
(defn keep
|
||||||
``Given a predicate `pred`, return a new array containing the truthy results
|
```
|
||||||
of applying `pred` to each element in the indexed collection `ind`. This is
|
Given a predicate `pred`, return a new array containing the
|
||||||
different from `filter` which returns an array of the original elements where
|
truthy results of applying `pred` to each value in the data
|
||||||
the predicate is truthy.``
|
structure `ind`, but only if no `inds` are provided. Multiple
|
||||||
|
data structures can be handled if each `inds` is a data
|
||||||
|
structure and `pred` is a function of arity one more than the
|
||||||
|
number of `inds`. The resulting array has a length that is no
|
||||||
|
longer than the shortest of `ind` and each of `inds`.
|
||||||
|
```
|
||||||
[pred ind & inds]
|
[pred ind & inds]
|
||||||
(def res @[])
|
(def res @[])
|
||||||
(map-template :keep res pred ind inds)
|
(map-template :keep res pred ind inds)
|
||||||
@@ -1311,7 +1337,7 @@
|
|||||||
(defdyn *redef* "When set, allow dynamically rebinding top level defs. Will slow generated code and is intended to be used for development.")
|
(defdyn *redef* "When set, allow dynamically rebinding top level defs. Will slow generated code and is intended to be used for development.")
|
||||||
(defdyn *debug* "Enables a built in debugger on errors and other useful features for debugging in a repl.")
|
(defdyn *debug* "Enables a built in debugger on errors and other useful features for debugging in a repl.")
|
||||||
(defdyn *exit* "When set, will cause the current context to complete. Can be set to exit from repl (or file), for example.")
|
(defdyn *exit* "When set, will cause the current context to complete. Can be set to exit from repl (or file), for example.")
|
||||||
(defdyn *exit-value* "Set the return value from `run-context` upon an exit. By default, `run-context` will return nil.")
|
(defdyn *exit-value* "Set the return value from `run-context` upon an exit.")
|
||||||
(defdyn *task-id* "When spawning a thread or fiber, the task-id can be assigned for concurrency control.")
|
(defdyn *task-id* "When spawning a thread or fiber, the task-id can be assigned for concurrency control.")
|
||||||
|
|
||||||
(defdyn *current-file*
|
(defdyn *current-file*
|
||||||
@@ -1908,7 +1934,7 @@
|
|||||||
that will match any value without creating a binding.
|
that will match any value without creating a binding.
|
||||||
|
|
||||||
While a symbol pattern will ordinarily match any value, the pattern `(@ <sym>)`,
|
While a symbol pattern will ordinarily match any value, the pattern `(@ <sym>)`,
|
||||||
where <sym> is any symbol, will attempt to match `x` against a value
|
where `<sym>` is any symbol, will attempt to match `x` against a value
|
||||||
already bound to `<sym>`, rather than matching and rebinding it.
|
already bound to `<sym>`, rather than matching and rebinding it.
|
||||||
|
|
||||||
Any other value pattern will only match if it is equal to `x`.
|
Any other value pattern will only match if it is equal to `x`.
|
||||||
@@ -2203,72 +2229,62 @@
|
|||||||
ret)
|
ret)
|
||||||
|
|
||||||
(defn all
|
(defn all
|
||||||
``Returns true if `(pred item)` is truthy for every item in `ind`.
|
```
|
||||||
Otherwise, returns the first falsey result encountered.
|
Returns true if applying `pred` to every value in a data
|
||||||
Returns true if `ind` is empty.``
|
structure `ind` results in only truthy values, but only if no
|
||||||
|
`inds` are provided. Multiple data structures can be handled
|
||||||
|
if each `inds` is a data structure and `pred` is a function
|
||||||
|
of arity one more than the number of `inds`. Returns the first
|
||||||
|
falsey result encountered. Note that `pred` is only called as
|
||||||
|
many times as the length of the shortest of `ind` and each of
|
||||||
|
`inds`. If `ind` or any of `inds` are empty, returns true.
|
||||||
|
```
|
||||||
[pred ind & inds]
|
[pred ind & inds]
|
||||||
(var res true)
|
(var res true)
|
||||||
(map-template :all res pred ind inds)
|
(map-template :all res pred ind inds)
|
||||||
res)
|
res)
|
||||||
|
|
||||||
(defn some
|
(defn some
|
||||||
``Returns nil if `(pred item)` is false or nil for every item in `ind`.
|
```
|
||||||
Otherwise, returns the first truthy result encountered.``
|
Returns nil if applying `pred` to every value in a data
|
||||||
|
structure `ind` results in only falsey values, but only if no
|
||||||
|
`inds` are provided. Multiple data structures can be handled
|
||||||
|
if each `inds` is a data structure and `pred` is a function
|
||||||
|
of arity one more than the number of `inds`. Returns the first
|
||||||
|
truthy result encountered. Note that `pred` is only called as
|
||||||
|
many times as the length of the shortest of `ind` and each of
|
||||||
|
`inds`. If `ind` or any of `inds` are empty, returns nil.
|
||||||
|
```
|
||||||
[pred ind & inds]
|
[pred ind & inds]
|
||||||
(var res nil)
|
(var res nil)
|
||||||
(map-template :some res pred ind inds)
|
(map-template :some res pred ind inds)
|
||||||
res)
|
res)
|
||||||
|
|
||||||
(defn deep-not=
|
|
||||||
``Like `not=`, but mutable types (arrays, tables, buffers) are considered
|
|
||||||
equal if they have identical structure. Much slower than `not=`.``
|
|
||||||
[x y]
|
|
||||||
(def tx (type x))
|
|
||||||
(or
|
|
||||||
(not= tx (type y))
|
|
||||||
(case tx
|
|
||||||
:tuple (or (not= (length x) (length y))
|
|
||||||
(do
|
|
||||||
(var ret false)
|
|
||||||
(forv i 0 (length x)
|
|
||||||
(def xx (in x i))
|
|
||||||
(def yy (in y i))
|
|
||||||
(if (deep-not= xx yy)
|
|
||||||
(break (set ret true))))
|
|
||||||
ret))
|
|
||||||
:array (or (not= (length x) (length y))
|
|
||||||
(do
|
|
||||||
(var ret false)
|
|
||||||
(forv i 0 (length x)
|
|
||||||
(def xx (in x i))
|
|
||||||
(def yy (in y i))
|
|
||||||
(if (deep-not= xx yy)
|
|
||||||
(break (set ret true))))
|
|
||||||
ret))
|
|
||||||
:struct (deep-not= (kvs x) (kvs y))
|
|
||||||
:table (deep-not= (table/to-struct x) (table/to-struct y))
|
|
||||||
:buffer (not= (string x) (string y))
|
|
||||||
(not= x y))))
|
|
||||||
|
|
||||||
(defn deep=
|
|
||||||
``Like `=`, but mutable types (arrays, tables, buffers) are considered
|
|
||||||
equal if they have identical structure. Much slower than `=`.``
|
|
||||||
[x y]
|
|
||||||
(not (deep-not= x y)))
|
|
||||||
|
|
||||||
(defn freeze
|
(defn freeze
|
||||||
`Freeze an object (make it immutable) and do a deep copy, making
|
`Freeze an object (make it immutable) and do a deep copy, making
|
||||||
child values also immutable. Closures, fibers, and abstract types
|
child values also immutable. Closures, fibers, and abstract types
|
||||||
will not be recursively frozen, but all other types will.`
|
will not be recursively frozen, but all other types will.`
|
||||||
[x]
|
[x]
|
||||||
(case (type x)
|
(def tx (type x))
|
||||||
:array (tuple/slice (map freeze x))
|
(cond
|
||||||
:tuple (tuple/slice (map freeze x))
|
(or (= tx :array) (= tx :tuple))
|
||||||
:table (if-let [p (table/getproto x)]
|
(tuple/slice (map freeze x))
|
||||||
(freeze (merge (table/clone p) x))
|
|
||||||
(struct ;(map freeze (kvs x))))
|
(or (= tx :table) (= tx :struct))
|
||||||
:struct (struct ;(map freeze (kvs x)))
|
(let [temp-tab @{}]
|
||||||
:buffer (string x)
|
# Handle multiple unique keys that freeze. Result should
|
||||||
|
# be independent of iteration order.
|
||||||
|
(eachp [k v] x
|
||||||
|
(def kk (freeze k))
|
||||||
|
(def vv (freeze v))
|
||||||
|
(def old (get temp-tab kk))
|
||||||
|
(def new (if (= nil old) vv (max vv old)))
|
||||||
|
(put temp-tab kk new))
|
||||||
|
(table/to-struct temp-tab (freeze (getproto x))))
|
||||||
|
|
||||||
|
(= tx :buffer)
|
||||||
|
(string x)
|
||||||
|
|
||||||
x))
|
x))
|
||||||
|
|
||||||
(defn thaw
|
(defn thaw
|
||||||
@@ -2284,6 +2300,41 @@
|
|||||||
:string (buffer ds)
|
:string (buffer ds)
|
||||||
ds))
|
ds))
|
||||||
|
|
||||||
|
(defn deep-not=
|
||||||
|
``Like `not=`, but mutable types (arrays, tables, buffers) are considered
|
||||||
|
equal if they have identical structure. Much slower than `not=`.``
|
||||||
|
[x y]
|
||||||
|
(def tx (type x))
|
||||||
|
(or
|
||||||
|
(not= tx (type y))
|
||||||
|
(cond
|
||||||
|
(or (= tx :tuple) (= tx :array))
|
||||||
|
(or (not= (length x) (length y))
|
||||||
|
(do
|
||||||
|
(var ret false)
|
||||||
|
(forv i 0 (length x)
|
||||||
|
(def xx (in x i))
|
||||||
|
(def yy (in y i))
|
||||||
|
(if (deep-not= xx yy)
|
||||||
|
(break (set ret true))))
|
||||||
|
ret))
|
||||||
|
(or (= tx :struct) (= tx :table))
|
||||||
|
(or (not= (length x) (length y))
|
||||||
|
(do
|
||||||
|
(def rawget (if (= tx :struct) struct/rawget table/rawget))
|
||||||
|
(var ret false)
|
||||||
|
(eachp [k v] x
|
||||||
|
(if (deep-not= (rawget y k) v) (break (set ret true))))
|
||||||
|
ret))
|
||||||
|
(= tx :buffer) (not= 0 (- (length x) (length y)) (memcmp x y))
|
||||||
|
(not= x y))))
|
||||||
|
|
||||||
|
(defn deep=
|
||||||
|
``Like `=`, but mutable types (arrays, tables, buffers) are considered
|
||||||
|
equal if they have identical structure. Much slower than `=`.``
|
||||||
|
[x y]
|
||||||
|
(not (deep-not= x y)))
|
||||||
|
|
||||||
(defn macex
|
(defn macex
|
||||||
``Expand macros completely.
|
``Expand macros completely.
|
||||||
`on-binding` is an optional callback for whenever a normal symbolic binding
|
`on-binding` is an optional callback for whenever a normal symbolic binding
|
||||||
@@ -2335,17 +2386,11 @@
|
|||||||
|
|
||||||
(defmacro short-fn
|
(defmacro short-fn
|
||||||
```
|
```
|
||||||
Shorthand for `fn`. Arguments are given as `$n`, where `n` is the 0-indexed
|
Shorthand for `fn`. Arguments are given as `$n`, where `n` is the
|
||||||
argument of the function. `$` is also an alias for the first (index 0) argument.
|
0-indexed argument of the function. `$` is also an alias for the
|
||||||
The `$&` symbol will make the anonymous function variadic if it appears in the
|
first (index 0) argument. The `$&` symbol will make the anonymous
|
||||||
body of the function, and can be combined with positional arguments.
|
function variadic if it appears in the body of the function, and
|
||||||
|
can be combined with positional arguments.
|
||||||
Example usage:
|
|
||||||
|
|
||||||
(short-fn (+ $ $)) # A function that doubles its arguments.
|
|
||||||
(short-fn (string $0 $1)) # accepting multiple args.
|
|
||||||
|(+ $ $) # use pipe reader macro for terse function literals.
|
|
||||||
|(+ $&) # variadic functions
|
|
||||||
```
|
```
|
||||||
[arg &opt name]
|
[arg &opt name]
|
||||||
(var max-param-seen -1)
|
(var max-param-seen -1)
|
||||||
@@ -2530,7 +2575,7 @@
|
|||||||
* `:env` -- the environment to compile against - default is the current env
|
* `:env` -- the environment to compile against - default is the current env
|
||||||
|
|
||||||
* `:source` -- source path for better errors (use keywords for non-paths) - default
|
* `:source` -- source path for better errors (use keywords for non-paths) - default
|
||||||
is :<anonymous>
|
is `:<anonymous>`
|
||||||
|
|
||||||
* `:on-compile-error` -- callback when compilation fails - default is bad-compile
|
* `:on-compile-error` -- callback when compilation fails - default is bad-compile
|
||||||
|
|
||||||
@@ -2854,8 +2899,8 @@
|
|||||||
(when (and (string? pattern) (string/has-prefix? ":sys:/" pattern))
|
(when (and (string? pattern) (string/has-prefix? ":sys:/" pattern))
|
||||||
(set last-index index)
|
(set last-index index)
|
||||||
(array/push copies [(string/replace ":sys:" path pattern) ;(drop 1 entry)])))
|
(array/push copies [(string/replace ":sys:" path pattern) ;(drop 1 entry)])))
|
||||||
(array/insert mp (+ 1 last-index) ;copies)
|
(array/insert mp (+ 1 last-index) ;copies)
|
||||||
mp)
|
mp)
|
||||||
|
|
||||||
(module/add-paths ":native:" :native)
|
(module/add-paths ":native:" :native)
|
||||||
(module/add-paths "/init.janet" :source)
|
(module/add-paths "/init.janet" :source)
|
||||||
@@ -3136,12 +3181,17 @@
|
|||||||
use the name of the module as a prefix. One can also use "`:export true`"
|
use the name of the module as a prefix. One can also use "`:export true`"
|
||||||
to re-export the imported symbols. If "`:exit true`" is given as an argument,
|
to re-export the imported symbols. If "`:exit true`" is given as an argument,
|
||||||
any errors encountered at the top level in the module will cause `(os/exit 1)`
|
any errors encountered at the top level in the module will cause `(os/exit 1)`
|
||||||
to be called. Dynamic bindings will NOT be imported. Use :fresh to bypass the
|
to be called. Dynamic bindings will NOT be imported. Use :fresh with a truthy
|
||||||
module cache. Use `:only [foo bar baz]` to only import select bindings into the
|
value to bypass the module cache. Use `:only [foo bar baz]` to only import
|
||||||
current environment.``
|
select bindings into the current environment.``
|
||||||
[path & args]
|
[path & args]
|
||||||
|
(assertf (even? (length args)) "args should have even length: %n" args)
|
||||||
(def ps (partition 2 args))
|
(def ps (partition 2 args))
|
||||||
(def argm (mapcat (fn [[k v]] [k (case k :as (string v) :only ~(quote ,v) v)]) ps))
|
(def argm
|
||||||
|
(mapcat (fn [[k v]]
|
||||||
|
(assertf (keyword? k) "expected keyword, got %s: %n" (type k) k)
|
||||||
|
[k (case k :as (string v) :only ~(quote ,v) v)])
|
||||||
|
ps))
|
||||||
(tuple import* (string path) ;argm))
|
(tuple import* (string path) ;argm))
|
||||||
|
|
||||||
(defmacro use
|
(defmacro use
|
||||||
@@ -3204,12 +3254,10 @@
|
|||||||
# Terminal codes for emission/tokenization
|
# Terminal codes for emission/tokenization
|
||||||
(def delimiters
|
(def delimiters
|
||||||
(if has-color
|
(if has-color
|
||||||
{:underline ["\e[4m" "\e[24m"]
|
{:code ["\e[97m" "\e[39m"]
|
||||||
:code ["\e[97m" "\e[39m"]
|
|
||||||
:italics ["\e[4m" "\e[24m"]
|
:italics ["\e[4m" "\e[24m"]
|
||||||
:bold ["\e[1m" "\e[22m"]}
|
:bold ["\e[1m" "\e[22m"]}
|
||||||
{:underline ["_" "_"]
|
{:code ["`" "`"]
|
||||||
:code ["`" "`"]
|
|
||||||
:italics ["*" "*"]
|
:italics ["*" "*"]
|
||||||
:bold ["**" "**"]}))
|
:bold ["**" "**"]}))
|
||||||
(def modes @{})
|
(def modes @{})
|
||||||
@@ -3340,7 +3388,6 @@
|
|||||||
(= b (chr `\`)) (do
|
(= b (chr `\`)) (do
|
||||||
(++ token-length)
|
(++ token-length)
|
||||||
(buffer/push token (get line (++ i))))
|
(buffer/push token (get line (++ i))))
|
||||||
(= b (chr "_")) (delim :underline)
|
|
||||||
(= b (chr "*"))
|
(= b (chr "*"))
|
||||||
(if (= (chr "*") (get line (+ i 1)))
|
(if (= (chr "*") (get line (+ i 1)))
|
||||||
(do (++ i)
|
(do (++ i)
|
||||||
@@ -3872,9 +3919,15 @@
|
|||||||
|
|
||||||
(compwhen (dyn 'net/listen)
|
(compwhen (dyn 'net/listen)
|
||||||
(defn net/server
|
(defn net/server
|
||||||
"Start a server asynchronously with `net/listen` and `net/accept-loop`. Returns the new server stream."
|
``
|
||||||
[host port &opt handler type]
|
Starts a server with `net/listen`. Runs `net/accept-loop` asynchronously if
|
||||||
(def s (net/listen host port type))
|
`handler` is set and `type` is `:stream` (the default). It is invalid to set
|
||||||
|
`handler` if `type` is `:datagram`. Returns the new server stream.
|
||||||
|
``
|
||||||
|
[host port &opt handler type no-reuse]
|
||||||
|
(assert (not (and (= type :datagram) handler))
|
||||||
|
"handler not supported for :datagram servers")
|
||||||
|
(def s (net/listen host port type no-reuse))
|
||||||
(if handler
|
(if handler
|
||||||
(ev/go (fn [] (net/accept-loop s handler))))
|
(ev/go (fn [] (net/accept-loop s handler))))
|
||||||
s))
|
s))
|
||||||
@@ -3980,7 +4033,7 @@
|
|||||||
|
|
||||||
(def- safe-forms {'defn true 'varfn true 'defn- true 'defmacro true 'defmacro- true
|
(def- safe-forms {'defn true 'varfn true 'defn- true 'defmacro true 'defmacro- true
|
||||||
'def is-safe-def 'var is-safe-def 'def- is-safe-def 'var- is-safe-def
|
'def is-safe-def 'var is-safe-def 'def- is-safe-def 'var- is-safe-def
|
||||||
'defglobal is-safe-def 'varglobal is-safe-def})
|
'defglobal is-safe-def 'varglobal is-safe-def 'defdyn true})
|
||||||
|
|
||||||
(def- importers {'import true 'import* true 'dofile true 'require true})
|
(def- importers {'import true 'import* true 'dofile true 'require true})
|
||||||
(defn- use-2 [evaluator args]
|
(defn- use-2 [evaluator args]
|
||||||
@@ -4096,7 +4149,7 @@
|
|||||||
(when (empty? b) (buffer/trim b) (os/chmod to perm) (break))
|
(when (empty? b) (buffer/trim b) (os/chmod to perm) (break))
|
||||||
(file/write fto b)
|
(file/write fto b)
|
||||||
(buffer/clear b)))
|
(buffer/clear b)))
|
||||||
(errorf "destination file %s cannot be opened for writing" to))
|
(errorf "destination file %s cannot be opened for writing" to))
|
||||||
(errorf "source file %s cannot be opened for reading" from)))
|
(errorf "source file %s cannot be opened for reading" from)))
|
||||||
|
|
||||||
(defn- copyrf
|
(defn- copyrf
|
||||||
@@ -4114,7 +4167,11 @@
|
|||||||
[manifest]
|
[manifest]
|
||||||
(def bn (get manifest :name))
|
(def bn (get manifest :name))
|
||||||
(def manifest-name (get-manifest-filename bn))
|
(def manifest-name (get-manifest-filename bn))
|
||||||
(spit manifest-name (string/format "%j\n" manifest)))
|
(def b @"")
|
||||||
|
(buffer/format b "%j" manifest) # make sure it is valid jdn
|
||||||
|
(buffer/clear b)
|
||||||
|
(buffer/format b "%.99m\n" manifest)
|
||||||
|
(spit manifest-name b))
|
||||||
|
|
||||||
(defn bundle/manifest
|
(defn bundle/manifest
|
||||||
"Get the manifest for a give installed bundle"
|
"Get the manifest for a give installed bundle"
|
||||||
@@ -4133,7 +4190,7 @@
|
|||||||
(os/cd workdir)
|
(os/cd workdir)
|
||||||
([_] (print "cannot enter source directory " workdir " for bundle " bundle-name)))
|
([_] (print "cannot enter source directory " workdir " for bundle " bundle-name)))
|
||||||
(defer (os/cd dir)
|
(defer (os/cd dir)
|
||||||
(def new-env (make-env (curenv)))
|
(def new-env (make-env))
|
||||||
(put new-env *module-cache* @{})
|
(put new-env *module-cache* @{})
|
||||||
(put new-env *module-loading* @{})
|
(put new-env *module-loading* @{})
|
||||||
(put new-env *module-make-env* (fn make-bundle-env [&] (make-env new-env)))
|
(put new-env *module-make-env* (fn make-bundle-env [&] (make-env new-env)))
|
||||||
@@ -4148,7 +4205,6 @@
|
|||||||
[module bundle-name hook & args]
|
[module bundle-name hook & args]
|
||||||
(def hookf (module/value module (symbol hook)))
|
(def hookf (module/value module (symbol hook)))
|
||||||
(unless hookf (break))
|
(unless hookf (break))
|
||||||
(def manifest (bundle/manifest bundle-name))
|
|
||||||
(def dir (os/cwd))
|
(def dir (os/cwd))
|
||||||
(os/cd (get module :workdir "."))
|
(os/cd (get module :workdir "."))
|
||||||
(defer (os/cd dir)
|
(defer (os/cd dir)
|
||||||
@@ -4249,51 +4305,49 @@
|
|||||||
"Install a bundle from the local filesystem. The name of the bundle will be inferred from the bundle, or passed as a parameter :name in `config`."
|
"Install a bundle from the local filesystem. The name of the bundle will be inferred from the bundle, or passed as a parameter :name in `config`."
|
||||||
[path &keys config]
|
[path &keys config]
|
||||||
(def path (bundle-rpath path))
|
(def path (bundle-rpath path))
|
||||||
(def clean (get config :clean))
|
|
||||||
(def check (get config :check))
|
|
||||||
(def s (sep))
|
(def s (sep))
|
||||||
# Check meta file for dependencies and default name
|
# Detect bundle name
|
||||||
(def infofile-pre-1 (string path s "bundle" s "info.jdn"))
|
(def infofile-src1 (string path s "bundle" s "info.jdn"))
|
||||||
(def infofile-pre (if (fexists infofile-pre-1) infofile-pre-1 (string path s "info.jdn"))) # allow for alias
|
(def infofile-src2 (string path s "info.jdn"))
|
||||||
(var default-bundle-name nil)
|
(def infofile-src (cond (fexists infofile-src1) infofile-src1
|
||||||
(when (os/stat infofile-pre :mode)
|
(fexists infofile-src2) infofile-src2))
|
||||||
(def info (-> infofile-pre slurp parse))
|
(def info (-?> infofile-src slurp parse))
|
||||||
(def deps (get info :dependencies @[]))
|
(def bundle-name (get config :name (get info :name)))
|
||||||
(set default-bundle-name (get info :name))
|
|
||||||
(def missing (seq [d :in deps :when (not (bundle/installed? d))] (string d)))
|
|
||||||
(when (next missing) (errorf "missing dependencies %s" (string/join missing ", "))))
|
|
||||||
(def bundle-name (get config :name default-bundle-name))
|
|
||||||
(assertf bundle-name "unable to infer bundle name for %v, use :name argument" path)
|
(assertf bundle-name "unable to infer bundle name for %v, use :name argument" path)
|
||||||
(assertf (not (string/check-set "\\/" bundle-name))
|
(assertf (not (string/check-set "\\/" bundle-name))
|
||||||
"bundle name %v cannot contain path separators" bundle-name)
|
"bundle name %v cannot contain path separators" bundle-name)
|
||||||
(assert (next bundle-name) "cannot use empty bundle-name")
|
(assert (next bundle-name) "cannot use empty bundle-name")
|
||||||
(assert (not (fexists (get-manifest-filename bundle-name)))
|
(assertf (not (fexists (get-manifest-filename bundle-name)))
|
||||||
"bundle is already installed")
|
"bundle %v is already installed" bundle-name)
|
||||||
# Setup installed paths
|
# Setup installed paths
|
||||||
(prime-bundle-paths)
|
(prime-bundle-paths)
|
||||||
(os/mkdir (bundle-dir bundle-name))
|
(os/mkdir (bundle-dir bundle-name))
|
||||||
# Aliases for common bundle/ files
|
# Copy infofile
|
||||||
(def bundle.janet (string path s "bundle.janet"))
|
(def infofile-dest (bundle-file bundle-name "info.jdn"))
|
||||||
(when (fexists bundle.janet) (copyfile bundle.janet (bundle-file bundle-name "init.janet")))
|
(when infofile-src (copyfile infofile-src infofile-dest))
|
||||||
(when (fexists infofile-pre) (copyfile infofile-pre (bundle-file bundle-name "info.jdn")))
|
# Copy aliased initfile
|
||||||
|
(def initfile-alias (string path s "bundle.janet"))
|
||||||
|
(def initfile-dest (bundle-file bundle-name "init.janet"))
|
||||||
|
(when (fexists initfile-alias) (copyfile initfile-alias initfile-dest))
|
||||||
# Copy some files into the new location unconditionally
|
# Copy some files into the new location unconditionally
|
||||||
(def implicit-sources (string path s "bundle"))
|
(def implicit-sources (string path s "bundle"))
|
||||||
(when (= :directory (os/stat implicit-sources :mode))
|
(when (= :directory (os/stat implicit-sources :mode))
|
||||||
(copyrf implicit-sources (bundle-dir bundle-name)))
|
(copyrf implicit-sources (bundle-dir bundle-name)))
|
||||||
(def man @{:name bundle-name :local-source path :files @[]})
|
(def man @{:name bundle-name :local-source path :files @[]})
|
||||||
(merge-into man config)
|
(merge-into man config)
|
||||||
(def infofile (bundle-file bundle-name "info.jdn"))
|
|
||||||
(put man :auto-remove (get config :auto-remove))
|
|
||||||
(sync-manifest man)
|
(sync-manifest man)
|
||||||
(edefer (do (print "installation error, uninstalling") (bundle/uninstall bundle-name))
|
(edefer (do (print "installation error, uninstalling") (bundle/uninstall bundle-name))
|
||||||
(when (os/stat infofile :mode)
|
(when (os/stat infofile-dest :mode)
|
||||||
(def info (-> infofile slurp parse))
|
(def info (-> infofile-dest slurp parse))
|
||||||
(def deps (get info :dependencies @[]))
|
(def deps (seq [d :in (get info :dependencies @[])]
|
||||||
|
(string (if (dictionary? d) (get d :name) d))))
|
||||||
(def missing (filter (complement bundle/installed?) deps))
|
(def missing (filter (complement bundle/installed?) deps))
|
||||||
(when (next missing)
|
(when (next missing)
|
||||||
(error (string "missing dependencies " (string/join missing ", "))))
|
(error (string "missing dependencies " (string/join missing ", "))))
|
||||||
(put man :dependencies deps)
|
(put man :dependencies deps)
|
||||||
(put man :info info))
|
(put man :info info))
|
||||||
|
(def clean (get config :clean))
|
||||||
|
(def check (get config :check))
|
||||||
(def module (get-bundle-module bundle-name))
|
(def module (get-bundle-module bundle-name))
|
||||||
(def all-hooks (seq [[k v] :pairs module :when (symbol? k) :unless (get v :private)] (keyword k)))
|
(def all-hooks (seq [[k v] :pairs module :when (symbol? k) :unless (get v :private)] (keyword k)))
|
||||||
(put man :hooks all-hooks)
|
(put man :hooks all-hooks)
|
||||||
@@ -4307,6 +4361,9 @@
|
|||||||
(when check
|
(when check
|
||||||
(do-hook module bundle-name :check man)))
|
(do-hook module bundle-name :check man)))
|
||||||
(print "installed " bundle-name)
|
(print "installed " bundle-name)
|
||||||
|
(when (get man :has-bin-script)
|
||||||
|
(def binpath (string (dyn *syspath*) s "bin"))
|
||||||
|
(eprintf "executable scripts have been installed to %s" binpath))
|
||||||
bundle-name)
|
bundle-name)
|
||||||
|
|
||||||
(defn- bundle/pack
|
(defn- bundle/pack
|
||||||
@@ -4341,14 +4398,15 @@
|
|||||||
(spit install-hook b))
|
(spit install-hook b))
|
||||||
dest-dir)
|
dest-dir)
|
||||||
|
|
||||||
(defn bundle/reinstall
|
(defn bundle/replace
|
||||||
"Reinstall an existing bundle from the local source code."
|
"Reinstall an existing bundle from a new directory. Similar to bundle/reinstall,
|
||||||
[bundle-name &keys new-config]
|
but installs the replacement bundle from any directory. This is necesarry to replace a package without
|
||||||
|
breaking any dependencies."
|
||||||
|
[bundle-name path &keys new-config]
|
||||||
(def manifest (bundle/manifest bundle-name))
|
(def manifest (bundle/manifest bundle-name))
|
||||||
(def path (get manifest :local-source))
|
|
||||||
(def config (get manifest :config @{}))
|
(def config (get manifest :config @{}))
|
||||||
(def s (sep))
|
(def s (sep))
|
||||||
(assert (= :directory (os/stat path :mode)) "local source not available")
|
(assertf (= :directory (os/stat path :mode)) "local source %v not available" path)
|
||||||
(def backup-dir (string (dyn *syspath*) s bundle-name ".backup"))
|
(def backup-dir (string (dyn *syspath*) s bundle-name ".backup"))
|
||||||
(rmrf backup-dir)
|
(rmrf backup-dir)
|
||||||
(def backup-bundle-source (bundle/pack bundle-name backup-dir true))
|
(def backup-bundle-source (bundle/pack bundle-name backup-dir true))
|
||||||
@@ -4361,6 +4419,14 @@
|
|||||||
(rmrf backup-bundle-source)
|
(rmrf backup-bundle-source)
|
||||||
bundle-name)
|
bundle-name)
|
||||||
|
|
||||||
|
(defn bundle/reinstall
|
||||||
|
"Reinstall an existing bundle from the local source code."
|
||||||
|
[bundle-name &keys new-config]
|
||||||
|
(def manifest (bundle/manifest bundle-name))
|
||||||
|
(def path (get manifest :local-source))
|
||||||
|
(bundle/replace bundle-name path ;(kvs new-config))
|
||||||
|
bundle-name)
|
||||||
|
|
||||||
(defn bundle/add-directory
|
(defn bundle/add-directory
|
||||||
"Add a directory during the install process relative to `(dyn *syspath*)`"
|
"Add a directory during the install process relative to `(dyn *syspath*)`"
|
||||||
[manifest dest &opt chmod-mode]
|
[manifest dest &opt chmod-mode]
|
||||||
@@ -4408,7 +4474,7 @@
|
|||||||
|
|
||||||
(defn bundle/add
|
(defn bundle/add
|
||||||
"Add files and directories during a bundle install relative to `(dyn *syspath*)`.
|
"Add files and directories during a bundle install relative to `(dyn *syspath*)`.
|
||||||
Added paths will be recorded in the bundle manifest such that they are properly tracked
|
Added files and directories will be recorded in the bundle manifest such that they are properly tracked
|
||||||
and removed during an upgrade or uninstall."
|
and removed during an upgrade or uninstall."
|
||||||
[manifest src &opt dest chmod-mode]
|
[manifest src &opt dest chmod-mode]
|
||||||
(default dest src)
|
(default dest src)
|
||||||
@@ -4424,13 +4490,17 @@
|
|||||||
(errorf "bad path %s - file is a %s" src mode)))
|
(errorf "bad path %s - file is a %s" src mode)))
|
||||||
|
|
||||||
(defn bundle/add-bin
|
(defn bundle/add-bin
|
||||||
`Shorthand for adding scripts during an install. Scripts will be installed to
|
``
|
||||||
(string (dyn *syspath*) "/bin") by default and will be set to be executable.`
|
Shorthand for adding scripts during an install. Scripts will be installed to
|
||||||
|
`(string (dyn *syspath*) "/bin")` by default and will be set to be executable.
|
||||||
|
``
|
||||||
[manifest src &opt dest chmod-mode]
|
[manifest src &opt dest chmod-mode]
|
||||||
(default dest (last (string/split "/" src)))
|
(def s (sep))
|
||||||
|
(default dest (last (string/split s src)))
|
||||||
(default chmod-mode 8r755)
|
(default chmod-mode 8r755)
|
||||||
(os/mkdir (string (dyn *syspath*) (sep) "bin"))
|
(os/mkdir (string (dyn *syspath*) s "bin"))
|
||||||
(bundle/add-file manifest src (string "bin" (sep) dest) chmod-mode))
|
(put manifest :has-bin-script true)
|
||||||
|
(bundle/add-file manifest src (string "bin" s dest) chmod-mode))
|
||||||
|
|
||||||
(defn bundle/update-all
|
(defn bundle/update-all
|
||||||
"Reinstall all bundles"
|
"Reinstall all bundles"
|
||||||
@@ -4493,6 +4563,12 @@
|
|||||||
"-nocolor" "n"
|
"-nocolor" "n"
|
||||||
"-color" "N"
|
"-color" "N"
|
||||||
"-library" "l"
|
"-library" "l"
|
||||||
|
"-install" "b"
|
||||||
|
"-reinstall" "B"
|
||||||
|
"-uninstall" "u"
|
||||||
|
"-update-all" "U"
|
||||||
|
"-list" "L"
|
||||||
|
"-prune" "P"
|
||||||
"-lint-warn" "w"
|
"-lint-warn" "w"
|
||||||
"-lint-error" "x"})
|
"-lint-error" "x"})
|
||||||
|
|
||||||
@@ -4503,7 +4579,7 @@
|
|||||||
|
|
||||||
(setdyn *args* args)
|
(setdyn *args* args)
|
||||||
|
|
||||||
(var should-repl false)
|
(var should-repl nil)
|
||||||
(var no-file true)
|
(var no-file true)
|
||||||
(var quiet false)
|
(var quiet false)
|
||||||
(var raw-stdin false)
|
(var raw-stdin false)
|
||||||
@@ -4558,6 +4634,12 @@
|
|||||||
--library (-l) lib : Use a module before processing more arguments
|
--library (-l) lib : Use a module before processing more arguments
|
||||||
--lint-warn (-w) level : Set the lint warning level - default is "normal"
|
--lint-warn (-w) level : Set the lint warning level - default is "normal"
|
||||||
--lint-error (-x) level : Set the lint error level - default is "none"
|
--lint-error (-x) level : Set the lint error level - default is "none"
|
||||||
|
--install (-b) dirpath : Install a bundle from a directory
|
||||||
|
--reinstall (-B) name : Reinstall a bundle by bundle name
|
||||||
|
--uninstall (-u) name : Uninstall a bundle by bundle name
|
||||||
|
--update-all (-U) : Reinstall all installed bundles
|
||||||
|
--prune (-P) : Uninstalled all bundles that are orphaned
|
||||||
|
--list (-L) : List all installed bundles
|
||||||
-- : Stop handling options
|
-- : Stop handling options
|
||||||
```)
|
```)
|
||||||
(os/exit 0)
|
(os/exit 0)
|
||||||
@@ -4602,6 +4684,30 @@
|
|||||||
((thunk) ;subargs)
|
((thunk) ;subargs)
|
||||||
(error (get thunk :error)))
|
(error (get thunk :error)))
|
||||||
math/inf)
|
math/inf)
|
||||||
|
"b"
|
||||||
|
(compif (dyn 'bundle/install)
|
||||||
|
(fn [i &] (bundle/install (in args (+ i 1))) (set no-file false) (if (= nil should-repl) (set should-repl false)) 2)
|
||||||
|
(fn [i &] (eprint "--install not supported with reduced os") 2))
|
||||||
|
"B"
|
||||||
|
(compif (dyn 'bundle/reinstall)
|
||||||
|
(fn [i &] (bundle/reinstall (in args (+ i 1))) (set no-file false) (if (= nil should-repl) (set should-repl false)) 2)
|
||||||
|
(fn [i &] (eprint "--reinstall not supported with reduced os") 2))
|
||||||
|
"u"
|
||||||
|
(compif (dyn 'bundle/uninstall)
|
||||||
|
(fn [i &] (bundle/uninstall (in args (+ i 1))) (set no-file false) (if (= nil should-repl) (set should-repl false)) 2)
|
||||||
|
(fn [i &] (eprint "--uninstall not supported with reduced os") 2))
|
||||||
|
"P"
|
||||||
|
(compif (dyn 'bundle/prune)
|
||||||
|
(fn [i &] (bundle/prune) (set no-file false) (if (= nil should-repl) (set should-repl false)) 1)
|
||||||
|
(fn [i &] (eprint "--prune not supported with reduced os") 1))
|
||||||
|
"U"
|
||||||
|
(compif (dyn 'bundle/update-all)
|
||||||
|
(fn [i &] (bundle/update-all) (set no-file false) (if (= nil should-repl) (set should-repl false)) 1)
|
||||||
|
(fn [i &] (eprint "--update-all not supported with reduced os") 1))
|
||||||
|
"L"
|
||||||
|
(compif (dyn 'bundle/list)
|
||||||
|
(fn [i &] (each l (bundle/list) (print l)) (set no-file false) (if (= nil should-repl) (set should-repl false)) 1)
|
||||||
|
(fn [i &] (eprint "--list not supported with reduced os") 1))
|
||||||
"d" (fn [&] (set debug-flag true) 1)
|
"d" (fn [&] (set debug-flag true) 1)
|
||||||
"w" (fn [i &] (set warn-level (get-lint-level i)) 2)
|
"w" (fn [i &] (set warn-level (get-lint-level i)) 2)
|
||||||
"x" (fn [i &] (set error-level (get-lint-level i)) 2)
|
"x" (fn [i &] (set error-level (get-lint-level i)) 2)
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -4,15 +4,16 @@
|
|||||||
#define JANETCONF_H
|
#define JANETCONF_H
|
||||||
|
|
||||||
#define JANET_VERSION_MAJOR 1
|
#define JANET_VERSION_MAJOR 1
|
||||||
#define JANET_VERSION_MINOR 37
|
#define JANET_VERSION_MINOR 39
|
||||||
#define JANET_VERSION_PATCH 0
|
#define JANET_VERSION_PATCH 0
|
||||||
#define JANET_VERSION_EXTRA ""
|
#define JANET_VERSION_EXTRA ""
|
||||||
#define JANET_VERSION "1.37.0"
|
#define JANET_VERSION "1.39.0"
|
||||||
|
|
||||||
/* #define JANET_BUILD "local" */
|
/* #define JANET_BUILD "local" */
|
||||||
|
|
||||||
/* These settings all affect linking, so use cautiously. */
|
/* These settings all affect linking, so use cautiously. */
|
||||||
/* #define JANET_SINGLE_THREADED */
|
/* #define JANET_SINGLE_THREADED */
|
||||||
|
/* #define JANET_THREAD_LOCAL _Thread_local */
|
||||||
/* #define JANET_NO_DYNAMIC_MODULES */
|
/* #define JANET_NO_DYNAMIC_MODULES */
|
||||||
/* #define JANET_NO_NANBOX */
|
/* #define JANET_NO_NANBOX */
|
||||||
/* #define JANET_API __attribute__((visibility ("default"))) */
|
/* #define JANET_API __attribute__((visibility ("default"))) */
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -62,6 +62,18 @@ JANET_NO_RETURN static void janet_top_level_signal(const char *msg) {
|
|||||||
|
|
||||||
void janet_signalv(JanetSignal sig, Janet message) {
|
void janet_signalv(JanetSignal sig, Janet message) {
|
||||||
if (janet_vm.return_reg != NULL) {
|
if (janet_vm.return_reg != NULL) {
|
||||||
|
/* Should match logic in janet_call for coercing everything not ok to an error (no awaits, yields, etc.) */
|
||||||
|
if (janet_vm.coerce_error && sig != JANET_SIGNAL_OK) {
|
||||||
|
#ifdef JANET_EV
|
||||||
|
if (NULL != janet_vm.root_fiber && sig == JANET_SIGNAL_EVENT) {
|
||||||
|
janet_vm.root_fiber->sched_id++;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
if (sig != JANET_SIGNAL_ERROR) {
|
||||||
|
message = janet_wrap_string(janet_formatc("%v coerced from %s to error", message, janet_signal_names[sig]));
|
||||||
|
}
|
||||||
|
sig = JANET_SIGNAL_ERROR;
|
||||||
|
}
|
||||||
*janet_vm.return_reg = message;
|
*janet_vm.return_reg = message;
|
||||||
if (NULL != janet_vm.fiber) {
|
if (NULL != janet_vm.fiber) {
|
||||||
janet_vm.fiber->flags |= JANET_FIBER_DID_LONGJUMP;
|
janet_vm.fiber->flags |= JANET_FIBER_DID_LONGJUMP;
|
||||||
@@ -577,6 +589,16 @@ JanetAtomicInt janet_atomic_load(JanetAtomicInt volatile *x) {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
JanetAtomicInt janet_atomic_load_relaxed(JanetAtomicInt volatile *x) {
|
||||||
|
#ifdef _MSC_VER
|
||||||
|
return _InterlockedOr(x, 0);
|
||||||
|
#elif defined(JANET_USE_STDATOMIC)
|
||||||
|
return atomic_load_explicit(x, memory_order_relaxed);
|
||||||
|
#else
|
||||||
|
return __atomic_load_n(x, __ATOMIC_RELAXED);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
/* Some definitions for function-like macros */
|
/* Some definitions for function-like macros */
|
||||||
|
|
||||||
JANET_API JanetStructHead *(janet_struct_head)(JanetStruct st) {
|
JANET_API JanetStructHead *(janet_struct_head)(JanetStruct st) {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -449,8 +449,9 @@ JANET_CORE_FN(janet_core_range,
|
|||||||
}
|
}
|
||||||
count = (count > 0) ? count : 0;
|
count = (count > 0) ? count : 0;
|
||||||
int32_t int_count;
|
int32_t int_count;
|
||||||
|
janet_assert(count >= 0, "bad range code");
|
||||||
if (count > (double) INT32_MAX) {
|
if (count > (double) INT32_MAX) {
|
||||||
int_count = INT32_MAX;
|
janet_panicf("range is too large, %f elements", count);
|
||||||
} else {
|
} else {
|
||||||
int_count = (int32_t) ceil(count);
|
int_count = (int32_t) ceil(count);
|
||||||
}
|
}
|
||||||
@@ -652,22 +653,15 @@ JANET_CORE_FN(janet_core_check_int,
|
|||||||
"(int? x)",
|
"(int? x)",
|
||||||
"Check if x can be exactly represented as a 32 bit signed two's complement integer.") {
|
"Check if x can be exactly represented as a 32 bit signed two's complement integer.") {
|
||||||
janet_fixarity(argc, 1);
|
janet_fixarity(argc, 1);
|
||||||
if (!janet_checktype(argv[0], JANET_NUMBER)) goto ret_false;
|
return janet_wrap_boolean(janet_checkint(argv[0]));
|
||||||
double num = janet_unwrap_number(argv[0]);
|
|
||||||
return janet_wrap_boolean(num == (double)((int32_t)num));
|
|
||||||
ret_false:
|
|
||||||
return janet_wrap_false();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(janet_core_check_nat,
|
JANET_CORE_FN(janet_core_check_nat,
|
||||||
"(nat? x)",
|
"(nat? x)",
|
||||||
"Check if x can be exactly represented as a non-negative 32 bit signed two's complement integer.") {
|
"Check if x can be exactly represented as a non-negative 32 bit signed two's complement integer.") {
|
||||||
janet_fixarity(argc, 1);
|
janet_fixarity(argc, 1);
|
||||||
if (!janet_checktype(argv[0], JANET_NUMBER)) goto ret_false;
|
if (!janet_checkint(argv[0])) return janet_wrap_false();
|
||||||
double num = janet_unwrap_number(argv[0]);
|
return janet_wrap_boolean(janet_unwrap_integer(argv[0]) >= 0);
|
||||||
return janet_wrap_boolean(num >= 0 && (num == (double)((int32_t)num)));
|
|
||||||
ret_false:
|
|
||||||
return janet_wrap_false();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(janet_core_is_bytes,
|
JANET_CORE_FN(janet_core_is_bytes,
|
||||||
@@ -1001,12 +995,11 @@ static void make_apply(JanetTable *env) {
|
|||||||
janet_quick_asm(env, JANET_FUN_APPLY | JANET_FUNCDEF_FLAG_VARARG,
|
janet_quick_asm(env, JANET_FUN_APPLY | JANET_FUNCDEF_FLAG_VARARG,
|
||||||
"apply", 1, 1, INT32_MAX, 6, apply_asm, sizeof(apply_asm),
|
"apply", 1, 1, INT32_MAX, 6, apply_asm, sizeof(apply_asm),
|
||||||
JDOC("(apply f & args)\n\n"
|
JDOC("(apply f & args)\n\n"
|
||||||
"Applies a function to a variable number of arguments. Each element in args "
|
"Applies a function f to a variable number of arguments. Each "
|
||||||
"is used as an argument to f, except the last element in args, which is expected to "
|
"element in args is used as an argument to f, except the last "
|
||||||
"be an array-like. Each element in this last argument is then also pushed as an argument to "
|
"element in args, which is expected to be an array or a tuple. "
|
||||||
"f. For example:\n\n"
|
"Each element in this last argument is then also pushed as an "
|
||||||
"\t(apply + 1000 (range 10))\n\n"
|
"argument to f."));
|
||||||
"sums the first 10 integers and 1000."));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static const uint32_t error_asm[] = {
|
static const uint32_t error_asm[] = {
|
||||||
@@ -1159,82 +1152,82 @@ JanetTable *janet_core_env(JanetTable *replacements) {
|
|||||||
janet_quick_asm(env, JANET_FUN_CMP,
|
janet_quick_asm(env, JANET_FUN_CMP,
|
||||||
"cmp", 2, 2, 2, 2, cmp_asm, sizeof(cmp_asm),
|
"cmp", 2, 2, 2, 2, cmp_asm, sizeof(cmp_asm),
|
||||||
JDOC("(cmp x y)\n\n"
|
JDOC("(cmp x y)\n\n"
|
||||||
"Returns -1 if x is strictly less than y, 1 if y is strictly greater "
|
"Returns -1 if x is strictly less than y, 1 if y is strictly greater "
|
||||||
"than x, and 0 otherwise. To return 0, x and y must be the exact same type."));
|
"than x, and 0 otherwise. To return 0, x and y must be the exact same type."));
|
||||||
janet_quick_asm(env, JANET_FUN_NEXT,
|
janet_quick_asm(env, JANET_FUN_NEXT,
|
||||||
"next", 2, 1, 2, 2, next_asm, sizeof(next_asm),
|
"next", 2, 1, 2, 2, next_asm, sizeof(next_asm),
|
||||||
JDOC("(next ds &opt key)\n\n"
|
JDOC("(next ds &opt key)\n\n"
|
||||||
"Gets the next key in a data structure. Can be used to iterate through "
|
"Gets the next key in a data structure. Can be used to iterate through "
|
||||||
"the keys of a data structure in an unspecified order. Keys are guaranteed "
|
"the keys of a data structure in an unspecified order. Keys are guaranteed "
|
||||||
"to be seen only once per iteration if the data structure is not mutated "
|
"to be seen only once per iteration if the data structure is not mutated "
|
||||||
"during iteration. If key is nil, next returns the first key. If next "
|
"during iteration. If key is nil, next returns the first key. If next "
|
||||||
"returns nil, there are no more keys to iterate through."));
|
"returns nil, there are no more keys to iterate through."));
|
||||||
janet_quick_asm(env, JANET_FUN_PROP,
|
janet_quick_asm(env, JANET_FUN_PROP,
|
||||||
"propagate", 2, 2, 2, 2, propagate_asm, sizeof(propagate_asm),
|
"propagate", 2, 2, 2, 2, propagate_asm, sizeof(propagate_asm),
|
||||||
JDOC("(propagate x fiber)\n\n"
|
JDOC("(propagate x fiber)\n\n"
|
||||||
"Propagate a signal from a fiber to the current fiber and "
|
"Propagate a signal from a fiber to the current fiber and "
|
||||||
"set the last value of the current fiber to `x`. The signal "
|
"set the last value of the current fiber to `x`. The signal "
|
||||||
"value is then available as the status of the current fiber. "
|
"value is then available as the status of the current fiber. "
|
||||||
"The resulting stack trace from the current fiber will include "
|
"The resulting stack trace from the current fiber will include "
|
||||||
"frames from fiber. If fiber is in a state that can be resumed, "
|
"frames from fiber. If fiber is in a state that can be resumed, "
|
||||||
"resuming the current fiber will first resume `fiber`. "
|
"resuming the current fiber will first resume `fiber`. "
|
||||||
"This function can be used to re-raise an error without losing "
|
"This function can be used to re-raise an error without losing "
|
||||||
"the original stack trace."));
|
"the original stack trace."));
|
||||||
janet_quick_asm(env, JANET_FUN_DEBUG,
|
janet_quick_asm(env, JANET_FUN_DEBUG,
|
||||||
"debug", 1, 0, 1, 1, debug_asm, sizeof(debug_asm),
|
"debug", 1, 0, 1, 1, debug_asm, sizeof(debug_asm),
|
||||||
JDOC("(debug &opt x)\n\n"
|
JDOC("(debug &opt x)\n\n"
|
||||||
"Throws a debug signal that can be caught by a parent fiber and used to inspect "
|
"Throws a debug signal that can be caught by a parent fiber and used to inspect "
|
||||||
"the running state of the current fiber. Returns the value passed in by resume."));
|
"the running state of the current fiber. Returns the value passed in by resume."));
|
||||||
janet_quick_asm(env, JANET_FUN_ERROR,
|
janet_quick_asm(env, JANET_FUN_ERROR,
|
||||||
"error", 1, 1, 1, 1, error_asm, sizeof(error_asm),
|
"error", 1, 1, 1, 1, error_asm, sizeof(error_asm),
|
||||||
JDOC("(error e)\n\n"
|
JDOC("(error e)\n\n"
|
||||||
"Throws an error e that can be caught and handled by a parent fiber."));
|
"Throws an error e that can be caught and handled by a parent fiber."));
|
||||||
janet_quick_asm(env, JANET_FUN_YIELD,
|
janet_quick_asm(env, JANET_FUN_YIELD,
|
||||||
"yield", 1, 0, 1, 2, yield_asm, sizeof(yield_asm),
|
"yield", 1, 0, 1, 2, yield_asm, sizeof(yield_asm),
|
||||||
JDOC("(yield &opt x)\n\n"
|
JDOC("(yield &opt x)\n\n"
|
||||||
"Yield a value to a parent fiber. When a fiber yields, its execution is paused until "
|
"Yield a value to a parent fiber. When a fiber yields, its execution is paused until "
|
||||||
"another thread resumes it. The fiber will then resume, and the last yield call will "
|
"another thread resumes it. The fiber will then resume, and the last yield call will "
|
||||||
"return the value that was passed to resume."));
|
"return the value that was passed to resume."));
|
||||||
janet_quick_asm(env, JANET_FUN_CANCEL,
|
janet_quick_asm(env, JANET_FUN_CANCEL,
|
||||||
"cancel", 2, 2, 2, 2, cancel_asm, sizeof(cancel_asm),
|
"cancel", 2, 2, 2, 2, cancel_asm, sizeof(cancel_asm),
|
||||||
JDOC("(cancel fiber err)\n\n"
|
JDOC("(cancel fiber err)\n\n"
|
||||||
"Resume a fiber but have it immediately raise an error. This lets a programmer unwind a pending fiber. "
|
"Resume a fiber but have it immediately raise an error. This lets a programmer unwind a pending fiber. "
|
||||||
"Returns the same result as resume."));
|
"Returns the same result as resume."));
|
||||||
janet_quick_asm(env, JANET_FUN_RESUME,
|
janet_quick_asm(env, JANET_FUN_RESUME,
|
||||||
"resume", 2, 1, 2, 2, resume_asm, sizeof(resume_asm),
|
"resume", 2, 1, 2, 2, resume_asm, sizeof(resume_asm),
|
||||||
JDOC("(resume fiber &opt x)\n\n"
|
JDOC("(resume fiber &opt x)\n\n"
|
||||||
"Resume a new or suspended fiber and optionally pass in a value to the fiber that "
|
"Resume a new or suspended fiber and optionally pass in a value to the fiber that "
|
||||||
"will be returned to the last yield in the case of a pending fiber, or the argument to "
|
"will be returned to the last yield in the case of a pending fiber, or the argument to "
|
||||||
"the dispatch function in the case of a new fiber. Returns either the return result of "
|
"the dispatch function in the case of a new fiber. Returns either the return result of "
|
||||||
"the fiber's dispatch function, or the value from the next yield call in fiber."));
|
"the fiber's dispatch function, or the value from the next yield call in fiber."));
|
||||||
janet_quick_asm(env, JANET_FUN_IN,
|
janet_quick_asm(env, JANET_FUN_IN,
|
||||||
"in", 3, 2, 3, 4, in_asm, sizeof(in_asm),
|
"in", 3, 2, 3, 4, in_asm, sizeof(in_asm),
|
||||||
JDOC("(in ds key &opt dflt)\n\n"
|
JDOC("(in ds key &opt dflt)\n\n"
|
||||||
"Get value in ds at key, works on associative data structures. Arrays, tuples, tables, structs, "
|
"Get value in ds at key, works on associative data structures. Arrays, tuples, tables, structs, "
|
||||||
"strings, symbols, and buffers are all associative and can be used. Arrays, tuples, strings, buffers, "
|
"strings, symbols, and buffers are all associative and can be used. Arrays, tuples, strings, buffers, "
|
||||||
"and symbols must use integer keys that are in bounds or an error is raised. Structs and tables can "
|
"and symbols must use integer keys that are in bounds or an error is raised. Structs and tables can "
|
||||||
"take any value as a key except nil and will return nil or dflt if not found."));
|
"take any value as a key except nil and will return nil or dflt if not found."));
|
||||||
janet_quick_asm(env, JANET_FUN_GET,
|
janet_quick_asm(env, JANET_FUN_GET,
|
||||||
"get", 3, 2, 3, 4, get_asm, sizeof(in_asm),
|
"get", 3, 2, 3, 4, get_asm, sizeof(in_asm),
|
||||||
JDOC("(get ds key &opt dflt)\n\n"
|
JDOC("(get ds key &opt dflt)\n\n"
|
||||||
"Get the value mapped to key in data structure ds, and return dflt or nil if not found. "
|
"Get the value mapped to key in data structure ds, and return dflt or nil if not found. "
|
||||||
"Similar to in, but will not throw an error if the key is invalid for the data structure "
|
"Similar to in, but will not throw an error if the key is invalid for the data structure "
|
||||||
"unless the data structure is an abstract type. In that case, the abstract type getter may throw "
|
"unless the data structure is an abstract type. In that case, the abstract type getter may throw "
|
||||||
"an error."));
|
"an error."));
|
||||||
janet_quick_asm(env, JANET_FUN_PUT,
|
janet_quick_asm(env, JANET_FUN_PUT,
|
||||||
"put", 3, 3, 3, 3, put_asm, sizeof(put_asm),
|
"put", 3, 3, 3, 3, put_asm, sizeof(put_asm),
|
||||||
JDOC("(put ds key value)\n\n"
|
JDOC("(put ds key value)\n\n"
|
||||||
"Associate a key with a value in any mutable associative data structure. Indexed data structures "
|
"Associate a key with a value in any mutable associative data structure. Indexed data structures "
|
||||||
"(arrays and buffers) only accept non-negative integer keys, and will expand if an out of bounds "
|
"(arrays and buffers) only accept non-negative integer keys, and will expand if an out of bounds "
|
||||||
"value is provided. In an array, extra space will be filled with nils, and in a buffer, extra "
|
"value is provided. In an array, extra space will be filled with nils, and in a buffer, extra "
|
||||||
"space will be filled with 0 bytes. In a table, putting a key that is contained in the table prototype "
|
"space will be filled with 0 bytes. In a table, putting a key that is contained in the table prototype "
|
||||||
"will hide the association defined by the prototype, but will not mutate the prototype table. Putting "
|
"will hide the association defined by the prototype, but will not mutate the prototype table. Putting "
|
||||||
"a value nil into a table will remove the key from the table. Returns the data structure ds."));
|
"a value nil into a table will remove the key from the table. Returns the data structure ds."));
|
||||||
janet_quick_asm(env, JANET_FUN_LENGTH,
|
janet_quick_asm(env, JANET_FUN_LENGTH,
|
||||||
"length", 1, 1, 1, 1, length_asm, sizeof(length_asm),
|
"length", 1, 1, 1, 1, length_asm, sizeof(length_asm),
|
||||||
JDOC("(length ds)\n\n"
|
JDOC("(length ds)\n\n"
|
||||||
"Returns the length or count of a data structure in constant time as an integer. For "
|
"Returns the length or count of a data structure in constant time as an integer. For "
|
||||||
"structs and tables, returns the number of key-value pairs in the data structure."));
|
"structs and tables, returns the number of key-value pairs in the data structure."));
|
||||||
janet_quick_asm(env, JANET_FUN_BNOT,
|
janet_quick_asm(env, JANET_FUN_BNOT,
|
||||||
"bnot", 1, 1, 1, 1, bnot_asm, sizeof(bnot_asm),
|
"bnot", 1, 1, 1, 1, bnot_asm, sizeof(bnot_asm),
|
||||||
JDOC("(bnot x)\n\nReturns the bit-wise inverse of integer x."));
|
JDOC("(bnot x)\n\nReturns the bit-wise inverse of integer x."));
|
||||||
@@ -1243,74 +1236,74 @@ JanetTable *janet_core_env(JanetTable *replacements) {
|
|||||||
/* Variadic ops */
|
/* Variadic ops */
|
||||||
templatize_varop(env, JANET_FUN_ADD, "+", 0, 0, JOP_ADD,
|
templatize_varop(env, JANET_FUN_ADD, "+", 0, 0, JOP_ADD,
|
||||||
JDOC("(+ & xs)\n\n"
|
JDOC("(+ & xs)\n\n"
|
||||||
"Returns the sum of all xs. xs must be integers or real numbers only. If xs is empty, return 0."));
|
"Returns the sum of all xs. xs must be integers or real numbers only. If xs is empty, return 0."));
|
||||||
templatize_varop(env, JANET_FUN_SUBTRACT, "-", 0, 0, JOP_SUBTRACT,
|
templatize_varop(env, JANET_FUN_SUBTRACT, "-", 0, 0, JOP_SUBTRACT,
|
||||||
JDOC("(- & xs)\n\n"
|
JDOC("(- & xs)\n\n"
|
||||||
"Returns the difference of xs. If xs is empty, returns 0. If xs has one element, returns the "
|
"Returns the difference of xs. If xs is empty, returns 0. If xs has one element, returns the "
|
||||||
"negative value of that element. Otherwise, returns the first element in xs minus the sum of "
|
"negative value of that element. Otherwise, returns the first element in xs minus the sum of "
|
||||||
"the rest of the elements."));
|
"the rest of the elements."));
|
||||||
templatize_varop(env, JANET_FUN_MULTIPLY, "*", 1, 1, JOP_MULTIPLY,
|
templatize_varop(env, JANET_FUN_MULTIPLY, "*", 1, 1, JOP_MULTIPLY,
|
||||||
JDOC("(* & xs)\n\n"
|
JDOC("(* & xs)\n\n"
|
||||||
"Returns the product of all elements in xs. If xs is empty, returns 1."));
|
"Returns the product of all elements in xs. If xs is empty, returns 1."));
|
||||||
templatize_varop(env, JANET_FUN_DIVIDE, "/", 1, 1, JOP_DIVIDE,
|
templatize_varop(env, JANET_FUN_DIVIDE, "/", 1, 1, JOP_DIVIDE,
|
||||||
JDOC("(/ & xs)\n\n"
|
JDOC("(/ & xs)\n\n"
|
||||||
"Returns the quotient of xs. If xs is empty, returns 1. If xs has one value x, returns "
|
"Returns the quotient of xs. If xs is empty, returns 1. If xs has one value x, returns "
|
||||||
"the reciprocal of x. Otherwise return the first value of xs repeatedly divided by the remaining "
|
"the reciprocal of x. Otherwise return the first value of xs repeatedly divided by the remaining "
|
||||||
"values."));
|
"values."));
|
||||||
templatize_varop(env, JANET_FUN_DIVIDE_FLOOR, "div", 1, 1, JOP_DIVIDE_FLOOR,
|
templatize_varop(env, JANET_FUN_DIVIDE_FLOOR, "div", 1, 1, JOP_DIVIDE_FLOOR,
|
||||||
JDOC("(div & xs)\n\n"
|
JDOC("(div & xs)\n\n"
|
||||||
"Returns the floored division of xs. If xs is empty, returns 1. If xs has one value x, returns "
|
"Returns the floored division of xs. If xs is empty, returns 1. If xs has one value x, returns "
|
||||||
"the reciprocal of x. Otherwise return the first value of xs repeatedly divided by the remaining "
|
"the reciprocal of x. Otherwise return the first value of xs repeatedly divided by the remaining "
|
||||||
"values."));
|
"values."));
|
||||||
templatize_varop(env, JANET_FUN_MODULO, "mod", 0, 1, JOP_MODULO,
|
templatize_varop(env, JANET_FUN_MODULO, "mod", 0, 1, JOP_MODULO,
|
||||||
JDOC("(mod & xs)\n\n"
|
JDOC("(mod & xs)\n\n"
|
||||||
"Returns the result of applying the modulo operator on the first value of xs with each remaining value. "
|
"Returns the result of applying the modulo operator on the first value of xs with each remaining value. "
|
||||||
"`(mod x 0)` is defined to be `x`."));
|
"`(mod x 0)` is defined to be `x`."));
|
||||||
templatize_varop(env, JANET_FUN_REMAINDER, "%", 0, 1, JOP_REMAINDER,
|
templatize_varop(env, JANET_FUN_REMAINDER, "%", 0, 1, JOP_REMAINDER,
|
||||||
JDOC("(% & xs)\n\n"
|
JDOC("(% & xs)\n\n"
|
||||||
"Returns the remainder of dividing the first value of xs by each remaining value."));
|
"Returns the remainder of dividing the first value of xs by each remaining value."));
|
||||||
templatize_varop(env, JANET_FUN_BAND, "band", -1, -1, JOP_BAND,
|
templatize_varop(env, JANET_FUN_BAND, "band", -1, -1, JOP_BAND,
|
||||||
JDOC("(band & xs)\n\n"
|
JDOC("(band & xs)\n\n"
|
||||||
"Returns the bit-wise and of all values in xs. Each x in xs must be an integer."));
|
"Returns the bit-wise and of all values in xs. Each x in xs must be an integer."));
|
||||||
templatize_varop(env, JANET_FUN_BOR, "bor", 0, 0, JOP_BOR,
|
templatize_varop(env, JANET_FUN_BOR, "bor", 0, 0, JOP_BOR,
|
||||||
JDOC("(bor & xs)\n\n"
|
JDOC("(bor & xs)\n\n"
|
||||||
"Returns the bit-wise or of all values in xs. Each x in xs must be an integer."));
|
"Returns the bit-wise or of all values in xs. Each x in xs must be an integer."));
|
||||||
templatize_varop(env, JANET_FUN_BXOR, "bxor", 0, 0, JOP_BXOR,
|
templatize_varop(env, JANET_FUN_BXOR, "bxor", 0, 0, JOP_BXOR,
|
||||||
JDOC("(bxor & xs)\n\n"
|
JDOC("(bxor & xs)\n\n"
|
||||||
"Returns the bit-wise xor of all values in xs. Each in xs must be an integer."));
|
"Returns the bit-wise xor of all values in xs. Each in xs must be an integer."));
|
||||||
templatize_varop(env, JANET_FUN_LSHIFT, "blshift", 1, 1, JOP_SHIFT_LEFT,
|
templatize_varop(env, JANET_FUN_LSHIFT, "blshift", 1, 1, JOP_SHIFT_LEFT,
|
||||||
JDOC("(blshift x & shifts)\n\n"
|
JDOC("(blshift x & shifts)\n\n"
|
||||||
"Returns the value of x bit shifted left by the sum of all values in shifts. x "
|
"Returns the value of x bit shifted left by the sum of all values in shifts. x "
|
||||||
"and each element in shift must be an integer."));
|
"and each element in shift must be an integer."));
|
||||||
templatize_varop(env, JANET_FUN_RSHIFT, "brshift", 1, 1, JOP_SHIFT_RIGHT,
|
templatize_varop(env, JANET_FUN_RSHIFT, "brshift", 1, 1, JOP_SHIFT_RIGHT,
|
||||||
JDOC("(brshift x & shifts)\n\n"
|
JDOC("(brshift x & shifts)\n\n"
|
||||||
"Returns the value of x bit shifted right by the sum of all values in shifts. x "
|
"Returns the value of x bit shifted right by the sum of all values in shifts. x "
|
||||||
"and each element in shift must be an integer."));
|
"and each element in shift must be an integer."));
|
||||||
templatize_varop(env, JANET_FUN_RSHIFTU, "brushift", 1, 1, JOP_SHIFT_RIGHT_UNSIGNED,
|
templatize_varop(env, JANET_FUN_RSHIFTU, "brushift", 1, 1, JOP_SHIFT_RIGHT_UNSIGNED,
|
||||||
JDOC("(brushift x & shifts)\n\n"
|
JDOC("(brushift x & shifts)\n\n"
|
||||||
"Returns the value of x bit shifted right by the sum of all values in shifts. x "
|
"Returns the value of x bit shifted right by the sum of all values in shifts. x "
|
||||||
"and each element in shift must be an integer. The sign of x is not preserved, so "
|
"and each element in shift must be an integer. The sign of x is not preserved, so "
|
||||||
"for positive shifts the return value will always be positive."));
|
"for positive shifts the return value will always be positive."));
|
||||||
|
|
||||||
/* Variadic comparators */
|
/* Variadic comparators */
|
||||||
templatize_comparator(env, JANET_FUN_GT, ">", 0, JOP_GREATER_THAN,
|
templatize_comparator(env, JANET_FUN_GT, ">", 0, JOP_GREATER_THAN,
|
||||||
JDOC("(> & xs)\n\n"
|
JDOC("(> & xs)\n\n"
|
||||||
"Check if xs is in descending order. Returns a boolean."));
|
"Check if xs is in descending order. Returns a boolean."));
|
||||||
templatize_comparator(env, JANET_FUN_LT, "<", 0, JOP_LESS_THAN,
|
templatize_comparator(env, JANET_FUN_LT, "<", 0, JOP_LESS_THAN,
|
||||||
JDOC("(< & xs)\n\n"
|
JDOC("(< & xs)\n\n"
|
||||||
"Check if xs is in ascending order. Returns a boolean."));
|
"Check if xs is in ascending order. Returns a boolean."));
|
||||||
templatize_comparator(env, JANET_FUN_GTE, ">=", 0, JOP_GREATER_THAN_EQUAL,
|
templatize_comparator(env, JANET_FUN_GTE, ">=", 0, JOP_GREATER_THAN_EQUAL,
|
||||||
JDOC("(>= & xs)\n\n"
|
JDOC("(>= & xs)\n\n"
|
||||||
"Check if xs is in non-ascending order. Returns a boolean."));
|
"Check if xs is in non-ascending order. Returns a boolean."));
|
||||||
templatize_comparator(env, JANET_FUN_LTE, "<=", 0, JOP_LESS_THAN_EQUAL,
|
templatize_comparator(env, JANET_FUN_LTE, "<=", 0, JOP_LESS_THAN_EQUAL,
|
||||||
JDOC("(<= & xs)\n\n"
|
JDOC("(<= & xs)\n\n"
|
||||||
"Check if xs is in non-descending order. Returns a boolean."));
|
"Check if xs is in non-descending order. Returns a boolean."));
|
||||||
templatize_comparator(env, JANET_FUN_EQ, "=", 0, JOP_EQUALS,
|
templatize_comparator(env, JANET_FUN_EQ, "=", 0, JOP_EQUALS,
|
||||||
JDOC("(= & xs)\n\n"
|
JDOC("(= & xs)\n\n"
|
||||||
"Check if all values in xs are equal. Returns a boolean."));
|
"Check if all values in xs are equal. Returns a boolean."));
|
||||||
templatize_comparator(env, JANET_FUN_NEQ, "not=", 1, JOP_EQUALS,
|
templatize_comparator(env, JANET_FUN_NEQ, "not=", 1, JOP_EQUALS,
|
||||||
JDOC("(not= & xs)\n\n"
|
JDOC("(not= & xs)\n\n"
|
||||||
"Check if any values in xs are not equal. Returns a boolean."));
|
"Check if any values in xs are not equal. Returns a boolean."));
|
||||||
|
|
||||||
/* Platform detection */
|
/* Platform detection */
|
||||||
janet_def(env, "janet/version", janet_cstringv(JANET_VERSION),
|
janet_def(env, "janet/version", janet_cstringv(JANET_VERSION),
|
||||||
@@ -1319,7 +1312,7 @@ JanetTable *janet_core_env(JanetTable *replacements) {
|
|||||||
JDOC("The build identifier of the running janet program."));
|
JDOC("The build identifier of the running janet program."));
|
||||||
janet_def(env, "janet/config-bits", janet_wrap_integer(JANET_CURRENT_CONFIG_BITS),
|
janet_def(env, "janet/config-bits", janet_wrap_integer(JANET_CURRENT_CONFIG_BITS),
|
||||||
JDOC("The flag set of config options from janetconf.h which is used to check "
|
JDOC("The flag set of config options from janetconf.h which is used to check "
|
||||||
"if native modules are compatible with the host program."));
|
"if native modules are compatible with the host program."));
|
||||||
|
|
||||||
/* Allow references to the environment */
|
/* Allow references to the environment */
|
||||||
janet_def(env, "root-env", janet_wrap_table(env),
|
janet_def(env, "root-env", janet_wrap_table(env),
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
254
src/core/ev.c
254
src/core/ev.c
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -32,9 +32,11 @@
|
|||||||
#ifdef JANET_EV
|
#ifdef JANET_EV
|
||||||
|
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
|
#include <fcntl.h>
|
||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
#include <winsock2.h>
|
#include <winsock2.h>
|
||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
|
#include <io.h>
|
||||||
#else
|
#else
|
||||||
#include <pthread.h>
|
#include <pthread.h>
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
@@ -43,7 +45,6 @@
|
|||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
#include <sys/ioctl.h>
|
#include <sys/ioctl.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <fcntl.h>
|
|
||||||
#include <netinet/in.h>
|
#include <netinet/in.h>
|
||||||
#include <netinet/tcp.h>
|
#include <netinet/tcp.h>
|
||||||
#include <netdb.h>
|
#include <netdb.h>
|
||||||
@@ -111,6 +112,13 @@ typedef struct {
|
|||||||
JanetHandle write_pipe;
|
JanetHandle write_pipe;
|
||||||
} JanetEVThreadInit;
|
} JanetEVThreadInit;
|
||||||
|
|
||||||
|
/* Structure used to initialize threads that run timeouts */
|
||||||
|
typedef struct {
|
||||||
|
double sec;
|
||||||
|
JanetVM *vm;
|
||||||
|
JanetFiber *fiber;
|
||||||
|
} JanetThreadedTimeout;
|
||||||
|
|
||||||
#define JANET_MAX_Q_CAPACITY 0x7FFFFFF
|
#define JANET_MAX_Q_CAPACITY 0x7FFFFFF
|
||||||
|
|
||||||
static void janet_q_init(JanetQueue *q) {
|
static void janet_q_init(JanetQueue *q) {
|
||||||
@@ -345,21 +353,22 @@ JanetStream *janet_stream(JanetHandle handle, uint32_t flags, const JanetMethod
|
|||||||
|
|
||||||
static void janet_stream_close_impl(JanetStream *stream) {
|
static void janet_stream_close_impl(JanetStream *stream) {
|
||||||
stream->flags |= JANET_STREAM_CLOSED;
|
stream->flags |= JANET_STREAM_CLOSED;
|
||||||
|
int canclose = !(stream->flags & JANET_STREAM_NOT_CLOSEABLE);
|
||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
if (stream->handle != INVALID_HANDLE_VALUE) {
|
if (stream->handle != INVALID_HANDLE_VALUE) {
|
||||||
#ifdef JANET_NET
|
#ifdef JANET_NET
|
||||||
if (stream->flags & JANET_STREAM_SOCKET) {
|
if (stream->flags & JANET_STREAM_SOCKET) {
|
||||||
closesocket((SOCKET) stream->handle);
|
if (canclose) closesocket((SOCKET) stream->handle);
|
||||||
} else
|
} else
|
||||||
#endif
|
#endif
|
||||||
{
|
{
|
||||||
CloseHandle(stream->handle);
|
if (canclose) CloseHandle(stream->handle);
|
||||||
}
|
}
|
||||||
stream->handle = INVALID_HANDLE_VALUE;
|
stream->handle = INVALID_HANDLE_VALUE;
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
if (stream->handle != -1) {
|
if (stream->handle != -1) {
|
||||||
close(stream->handle);
|
if (canclose) close(stream->handle);
|
||||||
stream->handle = -1;
|
stream->handle = -1;
|
||||||
#ifdef JANET_EV_POLL
|
#ifdef JANET_EV_POLL
|
||||||
uint32_t i = stream->index;
|
uint32_t i = stream->index;
|
||||||
@@ -595,8 +604,44 @@ void janet_ev_init_common(void) {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
static VOID CALLBACK janet_timeout_stop(ULONG_PTR ptr) {
|
||||||
|
UNREFERENCED_PARAMETER(ptr);
|
||||||
|
ExitThread(0);
|
||||||
|
}
|
||||||
|
#elif JANET_ANDROID
|
||||||
|
static void janet_timeout_stop(int sig_num) {
|
||||||
|
if (sig_num == SIGUSR1) {
|
||||||
|
pthread_exit(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static void handle_timeout_worker(JanetTimeout to, int cancel) {
|
||||||
|
if (!to.has_worker) return;
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
(void) cancel;
|
||||||
|
QueueUserAPC(janet_timeout_stop, to.worker, 0);
|
||||||
|
WaitForSingleObject(to.worker, INFINITE);
|
||||||
|
CloseHandle(to.worker);
|
||||||
|
#else
|
||||||
|
#ifdef JANET_ANDROID
|
||||||
|
if (cancel) janet_assert(!pthread_kill(to.worker, SIGUSR1), "pthread_kill");
|
||||||
|
#else
|
||||||
|
if (cancel) janet_assert(!pthread_cancel(to.worker), "pthread_cancel");
|
||||||
|
#endif
|
||||||
|
void *res = NULL;
|
||||||
|
janet_assert(!pthread_join(to.worker, &res), "pthread_join");
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
/* Common deinit code */
|
/* Common deinit code */
|
||||||
void janet_ev_deinit_common(void) {
|
void janet_ev_deinit_common(void) {
|
||||||
|
JanetTimeout to;
|
||||||
|
while (peek_timeout(&to)) {
|
||||||
|
handle_timeout_worker(to, 1);
|
||||||
|
pop_timeout(0);
|
||||||
|
}
|
||||||
janet_q_deinit(&janet_vm.spawn);
|
janet_q_deinit(&janet_vm.spawn);
|
||||||
janet_free(janet_vm.tq);
|
janet_free(janet_vm.tq);
|
||||||
janet_table_deinit(&janet_vm.threaded_abstracts);
|
janet_table_deinit(&janet_vm.threaded_abstracts);
|
||||||
@@ -622,6 +667,7 @@ void janet_addtimeout(double sec) {
|
|||||||
to.curr_fiber = NULL;
|
to.curr_fiber = NULL;
|
||||||
to.sched_id = fiber->sched_id;
|
to.sched_id = fiber->sched_id;
|
||||||
to.is_error = 1;
|
to.is_error = 1;
|
||||||
|
to.has_worker = 0;
|
||||||
add_timeout(to);
|
add_timeout(to);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -634,9 +680,51 @@ void janet_addtimeout_nil(double sec) {
|
|||||||
to.curr_fiber = NULL;
|
to.curr_fiber = NULL;
|
||||||
to.sched_id = fiber->sched_id;
|
to.sched_id = fiber->sched_id;
|
||||||
to.is_error = 0;
|
to.is_error = 0;
|
||||||
|
to.has_worker = 0;
|
||||||
add_timeout(to);
|
add_timeout(to);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void janet_timeout_cb(JanetEVGenericMessage msg) {
|
||||||
|
(void) msg;
|
||||||
|
janet_interpreter_interrupt_handled(&janet_vm);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
static DWORD WINAPI janet_timeout_body(LPVOID ptr) {
|
||||||
|
JanetThreadedTimeout tto = *(JanetThreadedTimeout *)ptr;
|
||||||
|
janet_free(ptr);
|
||||||
|
SleepEx((DWORD)(tto.sec * 1000), TRUE);
|
||||||
|
janet_interpreter_interrupt(tto.vm);
|
||||||
|
JanetEVGenericMessage msg = {0};
|
||||||
|
janet_ev_post_event(tto.vm, janet_timeout_cb, msg);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
static void *janet_timeout_body(void *ptr) {
|
||||||
|
#ifdef JANET_ANDROID
|
||||||
|
struct sigaction action;
|
||||||
|
memset(&action, 0, sizeof(action));
|
||||||
|
sigemptyset(&action.sa_mask);
|
||||||
|
action.sa_flags = 0;
|
||||||
|
action.sa_handler = &janet_timeout_stop;
|
||||||
|
sigaction(SIGUSR1, &action, NULL);
|
||||||
|
#endif
|
||||||
|
JanetThreadedTimeout tto = *(JanetThreadedTimeout *)ptr;
|
||||||
|
janet_free(ptr);
|
||||||
|
struct timespec ts;
|
||||||
|
ts.tv_sec = (time_t) tto.sec;
|
||||||
|
ts.tv_nsec = (tto.sec <= UINT32_MAX)
|
||||||
|
? (long)((tto.sec - ((uint32_t)tto.sec)) * 1000000000)
|
||||||
|
: 0;
|
||||||
|
nanosleep(&ts, &ts);
|
||||||
|
janet_interpreter_interrupt(tto.vm);
|
||||||
|
JanetEVGenericMessage msg = {0};
|
||||||
|
janet_ev_post_event(tto.vm, janet_timeout_cb, msg);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
void janet_ev_inc_refcount(void) {
|
void janet_ev_inc_refcount(void) {
|
||||||
janet_atomic_inc(&janet_vm.listener_count);
|
janet_atomic_inc(&janet_vm.listener_count);
|
||||||
}
|
}
|
||||||
@@ -1036,6 +1124,9 @@ JANET_CORE_FN(cfun_channel_push,
|
|||||||
"Returns the channel if the write succeeded, nil otherwise.") {
|
"Returns the channel if the write succeeded, nil otherwise.") {
|
||||||
janet_fixarity(argc, 2);
|
janet_fixarity(argc, 2);
|
||||||
JanetChannel *channel = janet_getchannel(argv, 0);
|
JanetChannel *channel = janet_getchannel(argv, 0);
|
||||||
|
if (janet_vm.coerce_error) {
|
||||||
|
janet_panic("cannot give to channel inside janet_call");
|
||||||
|
}
|
||||||
if (janet_channel_push(channel, argv[1], 0)) {
|
if (janet_channel_push(channel, argv[1], 0)) {
|
||||||
janet_await();
|
janet_await();
|
||||||
}
|
}
|
||||||
@@ -1048,6 +1139,9 @@ JANET_CORE_FN(cfun_channel_pop,
|
|||||||
janet_fixarity(argc, 1);
|
janet_fixarity(argc, 1);
|
||||||
JanetChannel *channel = janet_getchannel(argv, 0);
|
JanetChannel *channel = janet_getchannel(argv, 0);
|
||||||
Janet item;
|
Janet item;
|
||||||
|
if (janet_vm.coerce_error) {
|
||||||
|
janet_panic("cannot take from channel inside janet_call");
|
||||||
|
}
|
||||||
if (janet_channel_pop(channel, &item, 0)) {
|
if (janet_channel_pop(channel, &item, 0)) {
|
||||||
janet_schedule(janet_vm.root_fiber, item);
|
janet_schedule(janet_vm.root_fiber, item);
|
||||||
}
|
}
|
||||||
@@ -1084,6 +1178,10 @@ JANET_CORE_FN(cfun_channel_choice,
|
|||||||
int32_t len;
|
int32_t len;
|
||||||
const Janet *data;
|
const Janet *data;
|
||||||
|
|
||||||
|
if (janet_vm.coerce_error) {
|
||||||
|
janet_panic("cannot select from channel inside janet_call");
|
||||||
|
}
|
||||||
|
|
||||||
/* Check channels for immediate reads and writes */
|
/* Check channels for immediate reads and writes */
|
||||||
for (int32_t i = 0; i < argc; i++) {
|
for (int32_t i = 0; i < argc; i++) {
|
||||||
if (janet_indexed_view(argv[i], &data, &len) && len == 2) {
|
if (janet_indexed_view(argv[i], &data, &len) && len == 2) {
|
||||||
@@ -1373,12 +1471,13 @@ JanetFiber *janet_loop1(void) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
handle_timeout_worker(to, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Run scheduled fibers unless interrupts need to be handled. */
|
/* Run scheduled fibers unless interrupts need to be handled. */
|
||||||
while (janet_vm.spawn.head != janet_vm.spawn.tail) {
|
while (janet_vm.spawn.head != janet_vm.spawn.tail) {
|
||||||
/* Don't run until all interrupts have been marked as handled by calling janet_interpreter_interrupt_handled */
|
/* Don't run until all interrupts have been marked as handled by calling janet_interpreter_interrupt_handled */
|
||||||
if (janet_vm.auto_suspend) break;
|
if (janet_atomic_load_relaxed(&janet_vm.auto_suspend)) break;
|
||||||
JanetTask task = {NULL, janet_wrap_nil(), JANET_SIGNAL_OK, 0};
|
JanetTask task = {NULL, janet_wrap_nil(), JANET_SIGNAL_OK, 0};
|
||||||
janet_q_pop(&janet_vm.spawn, &task, sizeof(task));
|
janet_q_pop(&janet_vm.spawn, &task, sizeof(task));
|
||||||
if (task.fiber->gc.flags & JANET_FIBER_EV_FLAG_SUSPENDED) janet_ev_dec_refcount();
|
if (task.fiber->gc.flags & JANET_FIBER_EV_FLAG_SUSPENDED) janet_ev_dec_refcount();
|
||||||
@@ -1420,12 +1519,14 @@ JanetFiber *janet_loop1(void) {
|
|||||||
while ((has_timeout = peek_timeout(&to))) {
|
while ((has_timeout = peek_timeout(&to))) {
|
||||||
if (to.curr_fiber != NULL) {
|
if (to.curr_fiber != NULL) {
|
||||||
if (!janet_fiber_can_resume(to.curr_fiber)) {
|
if (!janet_fiber_can_resume(to.curr_fiber)) {
|
||||||
janet_table_remove(&janet_vm.active_tasks, janet_wrap_fiber(to.curr_fiber));
|
|
||||||
pop_timeout(0);
|
pop_timeout(0);
|
||||||
|
janet_table_remove(&janet_vm.active_tasks, janet_wrap_fiber(to.curr_fiber));
|
||||||
|
handle_timeout_worker(to, 1);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
} else if (to.fiber->sched_id != to.sched_id) {
|
} else if (to.fiber->sched_id != to.sched_id) {
|
||||||
pop_timeout(0);
|
pop_timeout(0);
|
||||||
|
handle_timeout_worker(to, 1);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -1590,7 +1691,7 @@ void janet_stream_level_triggered(JanetStream *stream) {
|
|||||||
|
|
||||||
static JanetTimestamp ts_now(void) {
|
static JanetTimestamp ts_now(void) {
|
||||||
struct timespec now;
|
struct timespec now;
|
||||||
janet_assert(-1 != clock_gettime(CLOCK_MONOTONIC, &now), "failed to get time");
|
janet_assert(-1 != janet_gettime(&now, JANET_TIME_MONOTONIC), "failed to get time");
|
||||||
uint64_t res = 1000 * now.tv_sec;
|
uint64_t res = 1000 * now.tv_sec;
|
||||||
res += now.tv_nsec / 1000000;
|
res += now.tv_nsec / 1000000;
|
||||||
return res;
|
return res;
|
||||||
@@ -1748,7 +1849,7 @@ JanetTimestamp to_interval(const JanetTimestamp ts) {
|
|||||||
|
|
||||||
static JanetTimestamp ts_now(void) {
|
static JanetTimestamp ts_now(void) {
|
||||||
struct timespec now;
|
struct timespec now;
|
||||||
janet_assert(-1 != clock_gettime(CLOCK_MONOTONIC, &now), "failed to get time");
|
janet_assert(-1 != janet_gettime(&now, JANET_TIME_MONOTONIC), "failed to get time");
|
||||||
uint64_t res = 1000 * now.tv_sec;
|
uint64_t res = 1000 * now.tv_sec;
|
||||||
res += now.tv_nsec / 1000000;
|
res += now.tv_nsec / 1000000;
|
||||||
return res;
|
return res;
|
||||||
@@ -1788,6 +1889,22 @@ void janet_stream_edge_triggered(JanetStream *stream) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void janet_stream_level_triggered(JanetStream *stream) {
|
void janet_stream_level_triggered(JanetStream *stream) {
|
||||||
|
/* On macos, we seem to need to delete any registered events before re-registering without
|
||||||
|
* EV_CLEAR, otherwise the new event will still have EV_CLEAR set erroneously. This could be a
|
||||||
|
* kernel bug, but unfortunately the specification is vague here, esp. in regards to where and when
|
||||||
|
* EV_CLEAR is set automatically. */
|
||||||
|
struct kevent kevs[2];
|
||||||
|
int length = 0;
|
||||||
|
if (stream->flags & (JANET_STREAM_READABLE | JANET_STREAM_ACCEPTABLE)) {
|
||||||
|
EV_SETx(&kevs[length++], stream->handle, EVFILT_READ, EV_DELETE, 0, 0, stream);
|
||||||
|
}
|
||||||
|
if (stream->flags & JANET_STREAM_WRITABLE) {
|
||||||
|
EV_SETx(&kevs[length++], stream->handle, EVFILT_WRITE, EV_DELETE, 0, 0, stream);
|
||||||
|
}
|
||||||
|
int status;
|
||||||
|
do {
|
||||||
|
status = kevent(janet_vm.kq, kevs, length, NULL, 0, NULL);
|
||||||
|
} while (status == -1 && errno == EINTR);
|
||||||
janet_register_stream_impl(stream, 0);
|
janet_register_stream_impl(stream, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1886,7 +2003,7 @@ void janet_ev_deinit(void) {
|
|||||||
|
|
||||||
static JanetTimestamp ts_now(void) {
|
static JanetTimestamp ts_now(void) {
|
||||||
struct timespec now;
|
struct timespec now;
|
||||||
janet_assert(-1 != clock_gettime(CLOCK_REALTIME, &now), "failed to get time");
|
janet_assert(-1 != janet_gettime(&now, JANET_TIME_MONOTONIC), "failed to get time");
|
||||||
uint64_t res = 1000 * now.tv_sec;
|
uint64_t res = 1000 * now.tv_sec;
|
||||||
res += now.tv_nsec / 1000000;
|
res += now.tv_nsec / 1000000;
|
||||||
return res;
|
return res;
|
||||||
@@ -2905,7 +3022,8 @@ static JanetEVGenericMessage janet_go_thread_subr(JanetEVGenericMessage args) {
|
|||||||
uint32_t count1;
|
uint32_t count1;
|
||||||
memcpy(&count1, nextbytes, sizeof(count1));
|
memcpy(&count1, nextbytes, sizeof(count1));
|
||||||
size_t count = (size_t) count1;
|
size_t count = (size_t) count1;
|
||||||
if (count > (endbytes - nextbytes) * sizeof(JanetCFunRegistry)) {
|
/* Use division to avoid overflowing size_t */
|
||||||
|
if (count > (endbytes - nextbytes - sizeof(count1)) / sizeof(JanetCFunRegistry)) {
|
||||||
janet_panic("thread message invalid");
|
janet_panic("thread message invalid");
|
||||||
}
|
}
|
||||||
janet_vm.registry_count = count;
|
janet_vm.registry_count = count;
|
||||||
@@ -3063,6 +3181,7 @@ JANET_NO_RETURN void janet_sleep_await(double sec) {
|
|||||||
to.is_error = 0;
|
to.is_error = 0;
|
||||||
to.sched_id = to.fiber->sched_id;
|
to.sched_id = to.fiber->sched_id;
|
||||||
to.curr_fiber = NULL;
|
to.curr_fiber = NULL;
|
||||||
|
to.has_worker = 0;
|
||||||
add_timeout(to);
|
add_timeout(to);
|
||||||
janet_await();
|
janet_await();
|
||||||
}
|
}
|
||||||
@@ -3076,26 +3195,56 @@ JANET_CORE_FN(cfun_ev_sleep,
|
|||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(cfun_ev_deadline,
|
JANET_CORE_FN(cfun_ev_deadline,
|
||||||
"(ev/deadline sec &opt tocancel tocheck)",
|
"(ev/deadline sec &opt tocancel tocheck intr?)",
|
||||||
"Schedules the event loop to try to cancel the `tocancel` "
|
"Schedules the event loop to try to cancel the `tocancel` task as with `ev/cancel`. "
|
||||||
"task as with `ev/cancel`. After `sec` seconds, the event "
|
"After `sec` seconds, the event loop will attempt cancellation of `tocancel` if the "
|
||||||
"loop will attempt cancellation of `tocancel` if the "
|
"`tocheck` fiber is resumable. `sec` is a number that can have a fractional part. "
|
||||||
"`tocheck` fiber is resumable. `sec` is a number that can "
|
"`tocancel` defaults to `(fiber/root)`, but if specified, must be a task (root "
|
||||||
"have a fractional part. `tocancel` defaults to "
|
"fiber). `tocheck` defaults to `(fiber/current)`, but if specified, must be a fiber. "
|
||||||
"`(fiber/root)`, but if specified, must be a task (root "
|
"Returns `tocancel` immediately. If `interrupt?` is set to true, will create a "
|
||||||
"fiber). `tocheck` defaults to `(fiber/current)`, but if "
|
"background thread to try to interrupt the VM if the timeout expires.") {
|
||||||
"specified, should be a fiber. Returns `tocancel` "
|
janet_arity(argc, 1, 4);
|
||||||
"immediately.") {
|
|
||||||
janet_arity(argc, 1, 3);
|
|
||||||
double sec = janet_getnumber(argv, 0);
|
double sec = janet_getnumber(argv, 0);
|
||||||
|
sec = (sec < 0) ? 0 : sec;
|
||||||
JanetFiber *tocancel = janet_optfiber(argv, argc, 1, janet_vm.root_fiber);
|
JanetFiber *tocancel = janet_optfiber(argv, argc, 1, janet_vm.root_fiber);
|
||||||
JanetFiber *tocheck = janet_optfiber(argv, argc, 2, janet_vm.fiber);
|
JanetFiber *tocheck = janet_optfiber(argv, argc, 2, janet_vm.fiber);
|
||||||
|
int use_interrupt = janet_optboolean(argv, argc, 3, 0);
|
||||||
JanetTimeout to;
|
JanetTimeout to;
|
||||||
to.when = ts_delta(ts_now(), sec);
|
to.when = ts_delta(ts_now(), sec);
|
||||||
to.fiber = tocancel;
|
to.fiber = tocancel;
|
||||||
to.curr_fiber = tocheck;
|
to.curr_fiber = tocheck;
|
||||||
to.is_error = 0;
|
to.is_error = 0;
|
||||||
to.sched_id = to.fiber->sched_id;
|
to.sched_id = to.fiber->sched_id;
|
||||||
|
if (use_interrupt) {
|
||||||
|
#ifdef JANET_ANDROID
|
||||||
|
janet_sandbox_assert(JANET_SANDBOX_SIGNAL);
|
||||||
|
#endif
|
||||||
|
JanetThreadedTimeout *tto = janet_malloc(sizeof(JanetThreadedTimeout));
|
||||||
|
if (NULL == tto) {
|
||||||
|
JANET_OUT_OF_MEMORY;
|
||||||
|
}
|
||||||
|
tto->sec = sec;
|
||||||
|
tto->vm = &janet_vm;
|
||||||
|
tto->fiber = tocheck;
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
HANDLE worker = CreateThread(NULL, 0, janet_timeout_body, tto, 0, NULL);
|
||||||
|
if (NULL == worker) {
|
||||||
|
janet_free(tto);
|
||||||
|
janet_panic("failed to create thread");
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
pthread_t worker;
|
||||||
|
int err = pthread_create(&worker, NULL, janet_timeout_body, tto);
|
||||||
|
if (err) {
|
||||||
|
janet_free(tto);
|
||||||
|
janet_panicf("%s", janet_strerror(err));
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
to.has_worker = 1;
|
||||||
|
to.worker = worker;
|
||||||
|
} else {
|
||||||
|
to.has_worker = 0;
|
||||||
|
}
|
||||||
add_timeout(to);
|
add_timeout(to);
|
||||||
return janet_wrap_fiber(tocancel);
|
return janet_wrap_fiber(tocancel);
|
||||||
}
|
}
|
||||||
@@ -3275,6 +3424,64 @@ JANET_CORE_FN(janet_cfun_rwlock_write_release,
|
|||||||
return argv[0];
|
return argv[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static JanetFile *get_file_for_stream(JanetStream *stream) {
|
||||||
|
int32_t flags = 0;
|
||||||
|
char fmt[4] = {0};
|
||||||
|
int index = 0;
|
||||||
|
if (stream->flags & JANET_STREAM_READABLE) {
|
||||||
|
flags |= JANET_FILE_READ;
|
||||||
|
janet_sandbox_assert(JANET_SANDBOX_FS_READ);
|
||||||
|
fmt[index++] = 'r';
|
||||||
|
}
|
||||||
|
if (stream->flags & JANET_STREAM_WRITABLE) {
|
||||||
|
flags |= JANET_FILE_WRITE;
|
||||||
|
janet_sandbox_assert(JANET_SANDBOX_FS_WRITE);
|
||||||
|
int currindex = index;
|
||||||
|
fmt[index++] = (currindex == 0) ? 'w' : '+';
|
||||||
|
}
|
||||||
|
if (index == 0) return NULL;
|
||||||
|
/* duplicate handle when converting stream to file */
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
int htype = 0;
|
||||||
|
if (fmt[0] == 'r' && fmt[1] == '+') {
|
||||||
|
htype = _O_RDWR;
|
||||||
|
} else if (fmt[0] == 'r') {
|
||||||
|
htype = _O_RDONLY;
|
||||||
|
} else if (fmt[0] == 'w') {
|
||||||
|
htype = _O_WRONLY;
|
||||||
|
}
|
||||||
|
int fd = _open_osfhandle((intptr_t) stream->handle, htype);
|
||||||
|
if (fd < 0) return NULL;
|
||||||
|
int fd_dup = _dup(fd);
|
||||||
|
if (fd_dup < 0) return NULL;
|
||||||
|
FILE *f = _fdopen(fd_dup, fmt);
|
||||||
|
if (NULL == f) {
|
||||||
|
_close(fd_dup);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
int fd_dup = dup(stream->handle);
|
||||||
|
if (fd_dup < 0) return NULL;
|
||||||
|
FILE *f = fdopen(fd_dup, fmt);
|
||||||
|
if (NULL == f) {
|
||||||
|
close(fd_dup);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
return janet_makejfile(f, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
JANET_CORE_FN(janet_cfun_to_file,
|
||||||
|
"(ev/to-file)",
|
||||||
|
"Create core/file copy of the stream. This value can be used "
|
||||||
|
"when blocking IO behavior is needed.") {
|
||||||
|
janet_fixarity(argc, 1);
|
||||||
|
JanetStream *stream = janet_getabstract(argv, 0, &janet_stream_type);
|
||||||
|
JanetFile *iof = get_file_for_stream(stream);
|
||||||
|
if (iof == NULL) janet_panic("cannot make file from stream");
|
||||||
|
return janet_wrap_abstract(iof);
|
||||||
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(janet_cfun_ev_all_tasks,
|
JANET_CORE_FN(janet_cfun_ev_all_tasks,
|
||||||
"(ev/all-tasks)",
|
"(ev/all-tasks)",
|
||||||
"Get an array of all active fibers that are being used by the scheduler.") {
|
"Get an array of all active fibers that are being used by the scheduler.") {
|
||||||
@@ -3319,6 +3526,7 @@ void janet_lib_ev(JanetTable *env) {
|
|||||||
JANET_CORE_REG("ev/acquire-wlock", janet_cfun_rwlock_write_lock),
|
JANET_CORE_REG("ev/acquire-wlock", janet_cfun_rwlock_write_lock),
|
||||||
JANET_CORE_REG("ev/release-rlock", janet_cfun_rwlock_read_release),
|
JANET_CORE_REG("ev/release-rlock", janet_cfun_rwlock_read_release),
|
||||||
JANET_CORE_REG("ev/release-wlock", janet_cfun_rwlock_write_release),
|
JANET_CORE_REG("ev/release-wlock", janet_cfun_rwlock_write_release),
|
||||||
|
JANET_CORE_REG("ev/to-file", janet_cfun_to_file),
|
||||||
JANET_CORE_REG("ev/all-tasks", janet_cfun_ev_all_tasks),
|
JANET_CORE_REG("ev/all-tasks", janet_cfun_ev_all_tasks),
|
||||||
JANET_REG_END
|
JANET_REG_END
|
||||||
};
|
};
|
||||||
@@ -3328,8 +3536,6 @@ void janet_lib_ev(JanetTable *env) {
|
|||||||
janet_register_abstract_type(&janet_channel_type);
|
janet_register_abstract_type(&janet_channel_type);
|
||||||
janet_register_abstract_type(&janet_mutex_type);
|
janet_register_abstract_type(&janet_mutex_type);
|
||||||
janet_register_abstract_type(&janet_rwlock_type);
|
janet_register_abstract_type(&janet_rwlock_type);
|
||||||
|
|
||||||
janet_lib_filewatch(env);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -1686,7 +1686,7 @@ JANET_CORE_FN(cfun_ffi_buffer_write,
|
|||||||
JanetFFIType type = decode_ffi_type(argv[0]);
|
JanetFFIType type = decode_ffi_type(argv[0]);
|
||||||
uint32_t el_size = (uint32_t) type_size(type);
|
uint32_t el_size = (uint32_t) type_size(type);
|
||||||
JanetBuffer *buffer = janet_optbuffer(argv, argc, 2, el_size);
|
JanetBuffer *buffer = janet_optbuffer(argv, argc, 2, el_size);
|
||||||
int32_t index = janet_optnat(argv, argc, 3, 0);
|
int32_t index = janet_optnat(argv, argc, 3, buffer->count);
|
||||||
int32_t old_count = buffer->count;
|
int32_t old_count = buffer->count;
|
||||||
if (index > old_count) janet_panic("index out of bounds");
|
if (index > old_count) janet_panic("index out of bounds");
|
||||||
buffer->count = index;
|
buffer->count = index;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -599,33 +599,33 @@ JANET_CORE_FN(cfun_filewatch_make,
|
|||||||
JANET_CORE_FN(cfun_filewatch_add,
|
JANET_CORE_FN(cfun_filewatch_add,
|
||||||
"(filewatch/add watcher path &opt flags)",
|
"(filewatch/add watcher path &opt flags)",
|
||||||
"Add a path to the watcher. Available flags depend on the current OS, and are as follows:\n\n"
|
"Add a path to the watcher. Available flags depend on the current OS, and are as follows:\n\n"
|
||||||
"Windows/MINGW (flags correspond to FILE_NOTIFY_CHANGE_* flags in win32 documentation):\n\n"
|
"Windows/MINGW (flags correspond to `FILE_NOTIFY_CHANGE_*` flags in win32 documentation):\n\n"
|
||||||
"* `:all` - trigger an event for all of the below triggers.\n\n"
|
"* `:all` - trigger an event for all of the below triggers.\n\n"
|
||||||
"* `:attributes` - FILE_NOTIFY_CHANGE_ATTRIBUTES\n\n"
|
"* `:attributes` - `FILE_NOTIFY_CHANGE_ATTRIBUTES`\n\n"
|
||||||
"* `:creation` - FILE_NOTIFY_CHANGE_CREATION\n\n"
|
"* `:creation` - `FILE_NOTIFY_CHANGE_CREATION`\n\n"
|
||||||
"* `:dir-name` - FILE_NOTIFY_CHANGE_DIR_NAME\n\n"
|
"* `:dir-name` - `FILE_NOTIFY_CHANGE_DIR_NAME`\n\n"
|
||||||
"* `:last-access` - FILE_NOTIFY_CHANGE_LAST_ACCESS\n\n"
|
"* `:last-access` - `FILE_NOTIFY_CHANGE_LAST_ACCESS`\n\n"
|
||||||
"* `:last-write` - FILE_NOTIFY_CHANGE_LAST_WRITE\n\n"
|
"* `:last-write` - `FILE_NOTIFY_CHANGE_LAST_WRITE`\n\n"
|
||||||
"* `:security` - FILE_NOTIFY_CHANGE_SECURITY\n\n"
|
"* `:security` - `FILE_NOTIFY_CHANGE_SECURITY`\n\n"
|
||||||
"* `:size` - FILE_NOTIFY_CHANGE_SIZE\n\n"
|
"* `:size` - `FILE_NOTIFY_CHANGE_SIZE`\n\n"
|
||||||
"* `:recursive` - watch subdirectories recursively\n\n"
|
"* `:recursive` - watch subdirectories recursively\n\n"
|
||||||
"Linux (flags correspond to IN_* flags from <sys/inotify.h>):\n\n"
|
"Linux (flags correspond to `IN_*` flags from <sys/inotify.h>):\n\n"
|
||||||
"* `:access` - IN_ACCESS\n\n"
|
"* `:access` - `IN_ACCESS`\n\n"
|
||||||
"* `:all` - IN_ALL_EVENTS\n\n"
|
"* `:all` - `IN_ALL_EVENTS`\n\n"
|
||||||
"* `:attrib` - IN_ATTRIB\n\n"
|
"* `:attrib` - `IN_ATTRIB`\n\n"
|
||||||
"* `:close-nowrite` - IN_CLOSE_NOWRITE\n\n"
|
"* `:close-nowrite` - `IN_CLOSE_NOWRITE`\n\n"
|
||||||
"* `:close-write` - IN_CLOSE_WRITE\n\n"
|
"* `:close-write` - `IN_CLOSE_WRITE`\n\n"
|
||||||
"* `:create` - IN_CREATE\n\n"
|
"* `:create` - `IN_CREATE`\n\n"
|
||||||
"* `:delete` - IN_DELETE\n\n"
|
"* `:delete` - `IN_DELETE`\n\n"
|
||||||
"* `:delete-self` - IN_DELETE_SELF\n\n"
|
"* `:delete-self` - `IN_DELETE_SELF`\n\n"
|
||||||
"* `:ignored` - IN_IGNORED\n\n"
|
"* `:ignored` - `IN_IGNORED`\n\n"
|
||||||
"* `:modify` - IN_MODIFY\n\n"
|
"* `:modify` - `IN_MODIFY`\n\n"
|
||||||
"* `:move-self` - IN_MOVE_SELF\n\n"
|
"* `:move-self` - `IN_MOVE_SELF`\n\n"
|
||||||
"* `:moved-from` - IN_MOVED_FROM\n\n"
|
"* `:moved-from` - `IN_MOVED_FROM`\n\n"
|
||||||
"* `:moved-to` - IN_MOVED_TO\n\n"
|
"* `:moved-to` - `IN_MOVED_TO`\n\n"
|
||||||
"* `:open` - IN_OPEN\n\n"
|
"* `:open` - `IN_OPEN`\n\n"
|
||||||
"* `:q-overflow` - IN_Q_OVERFLOW\n\n"
|
"* `:q-overflow` - `IN_Q_OVERFLOW`\n\n"
|
||||||
"* `:unmount` - IN_UNMOUNT\n\n\n"
|
"* `:unmount` - `IN_UNMOUNT`\n\n\n"
|
||||||
"On Windows, events will have the following possible types:\n\n"
|
"On Windows, events will have the following possible types:\n\n"
|
||||||
"* `:unknown`\n\n"
|
"* `:unknown`\n\n"
|
||||||
"* `:added`\n\n"
|
"* `:added`\n\n"
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose & contributors
|
* Copyright (c) 2025 Calvin Rose & contributors
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -191,21 +191,21 @@ Janet janet_wrap_u64(uint64_t x) {
|
|||||||
|
|
||||||
JANET_CORE_FN(cfun_it_s64_new,
|
JANET_CORE_FN(cfun_it_s64_new,
|
||||||
"(int/s64 value)",
|
"(int/s64 value)",
|
||||||
"Create a boxed signed 64 bit integer from a string value.") {
|
"Create a boxed signed 64 bit integer from a string value or a number.") {
|
||||||
janet_fixarity(argc, 1);
|
janet_fixarity(argc, 1);
|
||||||
return janet_wrap_s64(janet_unwrap_s64(argv[0]));
|
return janet_wrap_s64(janet_unwrap_s64(argv[0]));
|
||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(cfun_it_u64_new,
|
JANET_CORE_FN(cfun_it_u64_new,
|
||||||
"(int/u64 value)",
|
"(int/u64 value)",
|
||||||
"Create a boxed unsigned 64 bit integer from a string value.") {
|
"Create a boxed unsigned 64 bit integer from a string value or a number.") {
|
||||||
janet_fixarity(argc, 1);
|
janet_fixarity(argc, 1);
|
||||||
return janet_wrap_u64(janet_unwrap_u64(argv[0]));
|
return janet_wrap_u64(janet_unwrap_u64(argv[0]));
|
||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(cfun_to_number,
|
JANET_CORE_FN(cfun_to_number,
|
||||||
"(int/to-number value)",
|
"(int/to-number value)",
|
||||||
"Convert an int/u64 or int/s64 to a number. Fails if the number is out of range for an int32.") {
|
"Convert an int/u64 or int/s64 to a number. Fails if the number is out of range for an int64.") {
|
||||||
janet_fixarity(argc, 1);
|
janet_fixarity(argc, 1);
|
||||||
if (janet_type(argv[0]) == JANET_ABSTRACT) {
|
if (janet_type(argv[0]) == JANET_ABSTRACT) {
|
||||||
void *abst = janet_unwrap_abstract(argv[0]);
|
void *abst = janet_unwrap_abstract(argv[0]);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -31,6 +31,7 @@
|
|||||||
|
|
||||||
#ifndef JANET_WINDOWS
|
#ifndef JANET_WINDOWS
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
#include <sys/wait.h>
|
#include <sys/wait.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#endif
|
#endif
|
||||||
@@ -164,6 +165,14 @@ JANET_CORE_FN(cfun_io_fopen,
|
|||||||
}
|
}
|
||||||
FILE *f = fopen((const char *)fname, (const char *)fmode);
|
FILE *f = fopen((const char *)fname, (const char *)fmode);
|
||||||
if (f != NULL) {
|
if (f != NULL) {
|
||||||
|
#ifndef JANET_WINDOWS
|
||||||
|
struct stat st;
|
||||||
|
fstat(fileno(f), &st);
|
||||||
|
if (S_ISDIR(st.st_mode)) {
|
||||||
|
fclose(f);
|
||||||
|
janet_panicf("cannot open directory: %s", fname);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
size_t bufsize = janet_optsize(argv, argc, 2, BUFSIZ);
|
size_t bufsize = janet_optsize(argv, argc, 2, BUFSIZ);
|
||||||
if (bufsize != BUFSIZ) {
|
if (bufsize != BUFSIZ) {
|
||||||
int result = setvbuf(f, NULL, bufsize ? _IOFBF : _IONBF, bufsize);
|
int result = setvbuf(f, NULL, bufsize ? _IOFBF : _IONBF, bufsize);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
125
src/core/net.c
125
src/core/net.c
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose and contributors.
|
* Copyright (c) 2025 Calvin Rose and contributors.
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -341,7 +341,7 @@ static int janet_get_sockettype(Janet *argv, int32_t argc, int32_t n) {
|
|||||||
/* Needs argc >= offset + 2 */
|
/* Needs argc >= offset + 2 */
|
||||||
/* For unix paths, just rertuns a single sockaddr and sets *is_unix to 1,
|
/* For unix paths, just rertuns a single sockaddr and sets *is_unix to 1,
|
||||||
* otherwise 0. Also, ignores is_bind when is a unix socket. */
|
* otherwise 0. Also, ignores is_bind when is a unix socket. */
|
||||||
static struct addrinfo *janet_get_addrinfo(Janet *argv, int32_t offset, int socktype, int passive, int *is_unix) {
|
static struct addrinfo *janet_get_addrinfo(Janet *argv, int32_t offset, int socktype, int passive, int *is_unix, socklen_t *sizeout) {
|
||||||
/* Unix socket support - not yet supported on windows. */
|
/* Unix socket support - not yet supported on windows. */
|
||||||
#ifndef JANET_WINDOWS
|
#ifndef JANET_WINDOWS
|
||||||
if (janet_keyeq(argv[offset], "unix")) {
|
if (janet_keyeq(argv[offset], "unix")) {
|
||||||
@@ -352,15 +352,14 @@ static struct addrinfo *janet_get_addrinfo(Janet *argv, int32_t offset, int sock
|
|||||||
}
|
}
|
||||||
saddr->sun_family = AF_UNIX;
|
saddr->sun_family = AF_UNIX;
|
||||||
size_t path_size = sizeof(saddr->sun_path);
|
size_t path_size = sizeof(saddr->sun_path);
|
||||||
|
snprintf(saddr->sun_path, path_size, "%s", path);
|
||||||
|
*sizeout = sizeof(struct sockaddr_un);
|
||||||
#ifdef JANET_LINUX
|
#ifdef JANET_LINUX
|
||||||
if (path[0] == '@') {
|
if (path[0] == '@') {
|
||||||
saddr->sun_path[0] = '\0';
|
saddr->sun_path[0] = '\0';
|
||||||
snprintf(saddr->sun_path + 1, path_size - 1, "%s", path + 1);
|
*sizeout = offsetof(struct sockaddr_un, sun_path) + janet_string_length(path);
|
||||||
} else
|
|
||||||
#endif
|
|
||||||
{
|
|
||||||
snprintf(saddr->sun_path, path_size, "%s", path);
|
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
*is_unix = 1;
|
*is_unix = 1;
|
||||||
return (struct addrinfo *) saddr;
|
return (struct addrinfo *) saddr;
|
||||||
}
|
}
|
||||||
@@ -385,6 +384,11 @@ static struct addrinfo *janet_get_addrinfo(Janet *argv, int32_t offset, int sock
|
|||||||
janet_panicf("could not get address info: %s", gai_strerror(status));
|
janet_panicf("could not get address info: %s", gai_strerror(status));
|
||||||
}
|
}
|
||||||
*is_unix = 0;
|
*is_unix = 0;
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
*sizeout = 0;
|
||||||
|
#else
|
||||||
|
*sizeout = sizeof(struct sockaddr_un);
|
||||||
|
#endif
|
||||||
return ai;
|
return ai;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -405,12 +409,13 @@ JANET_CORE_FN(cfun_net_sockaddr,
|
|||||||
int socktype = janet_get_sockettype(argv, argc, 2);
|
int socktype = janet_get_sockettype(argv, argc, 2);
|
||||||
int is_unix = 0;
|
int is_unix = 0;
|
||||||
int make_arr = (argc >= 3 && janet_truthy(argv[3]));
|
int make_arr = (argc >= 3 && janet_truthy(argv[3]));
|
||||||
struct addrinfo *ai = janet_get_addrinfo(argv, 0, socktype, 0, &is_unix);
|
socklen_t addrsize = 0;
|
||||||
|
struct addrinfo *ai = janet_get_addrinfo(argv, 0, socktype, 0, &is_unix, &addrsize);
|
||||||
#ifndef JANET_WINDOWS
|
#ifndef JANET_WINDOWS
|
||||||
/* no unix domain socket support on windows yet */
|
/* no unix domain socket support on windows yet */
|
||||||
if (is_unix) {
|
if (is_unix) {
|
||||||
void *abst = janet_abstract(&janet_address_type, sizeof(struct sockaddr_un));
|
void *abst = janet_abstract(&janet_address_type, addrsize);
|
||||||
memcpy(abst, ai, sizeof(struct sockaddr_un));
|
memcpy(abst, ai, addrsize);
|
||||||
Janet ret = janet_wrap_abstract(abst);
|
Janet ret = janet_wrap_abstract(abst);
|
||||||
return make_arr ? janet_wrap_array(janet_array_n(&ret, 1)) : ret;
|
return make_arr ? janet_wrap_array(janet_array_n(&ret, 1)) : ret;
|
||||||
}
|
}
|
||||||
@@ -461,7 +466,8 @@ JANET_CORE_FN(cfun_net_connect,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Where we're connecting to */
|
/* Where we're connecting to */
|
||||||
struct addrinfo *ai = janet_get_addrinfo(argv, 0, socktype, 0, &is_unix);
|
socklen_t addrlen = 0;
|
||||||
|
struct addrinfo *ai = janet_get_addrinfo(argv, 0, socktype, 0, &is_unix, &addrlen);
|
||||||
|
|
||||||
/* Check if we're binding address */
|
/* Check if we're binding address */
|
||||||
struct addrinfo *binding = NULL;
|
struct addrinfo *binding = NULL;
|
||||||
@@ -486,7 +492,6 @@ JANET_CORE_FN(cfun_net_connect,
|
|||||||
/* Create socket */
|
/* Create socket */
|
||||||
JSock sock = JSOCKDEFAULT;
|
JSock sock = JSOCKDEFAULT;
|
||||||
void *addr = NULL;
|
void *addr = NULL;
|
||||||
socklen_t addrlen = 0;
|
|
||||||
#ifndef JANET_WINDOWS
|
#ifndef JANET_WINDOWS
|
||||||
if (is_unix) {
|
if (is_unix) {
|
||||||
sock = socket(AF_UNIX, socktype | JSOCKFLAGS, 0);
|
sock = socket(AF_UNIX, socktype | JSOCKFLAGS, 0);
|
||||||
@@ -496,7 +501,6 @@ JANET_CORE_FN(cfun_net_connect,
|
|||||||
janet_panicf("could not create socket: %V", v);
|
janet_panicf("could not create socket: %V", v);
|
||||||
}
|
}
|
||||||
addr = (void *) ai;
|
addr = (void *) ai;
|
||||||
addrlen = sizeof(struct sockaddr_un);
|
|
||||||
} else
|
} else
|
||||||
#endif
|
#endif
|
||||||
{
|
{
|
||||||
@@ -543,7 +547,9 @@ JANET_CORE_FN(cfun_net_connect,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Wrap socket in abstract type JanetStream */
|
/* Wrap socket in abstract type JanetStream */
|
||||||
JanetStream *stream = make_stream(sock, JANET_STREAM_READABLE | JANET_STREAM_WRITABLE);
|
uint32_t udp_flag = 0;
|
||||||
|
if (socktype == SOCK_DGRAM) udp_flag = JANET_STREAM_UDPSERVER;
|
||||||
|
JanetStream *stream = make_stream(sock, JANET_STREAM_READABLE | JANET_STREAM_WRITABLE | udp_flag);
|
||||||
|
|
||||||
/* Set up the socket for non-blocking IO before connecting */
|
/* Set up the socket for non-blocking IO before connecting */
|
||||||
janet_net_socknoblock(sock);
|
janet_net_socknoblock(sock);
|
||||||
@@ -554,7 +560,10 @@ JANET_CORE_FN(cfun_net_connect,
|
|||||||
int err = WSAGetLastError();
|
int err = WSAGetLastError();
|
||||||
freeaddrinfo(ai);
|
freeaddrinfo(ai);
|
||||||
#else
|
#else
|
||||||
int status = connect(sock, addr, addrlen);
|
int status;
|
||||||
|
do {
|
||||||
|
status = connect(sock, addr, addrlen);
|
||||||
|
} while (status == -1 && errno == EINTR);
|
||||||
int err = errno;
|
int err = errno;
|
||||||
if (is_unix) {
|
if (is_unix) {
|
||||||
janet_free(ai);
|
janet_free(ai);
|
||||||
@@ -578,17 +587,73 @@ JANET_CORE_FN(cfun_net_connect,
|
|||||||
net_sched_connect(stream);
|
net_sched_connect(stream);
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char *serverify_socket(JSock sfd) {
|
JANET_CORE_FN(cfun_net_socket,
|
||||||
|
"(net/socket &opt type)",
|
||||||
|
"Creates a new unbound socket. Type is an optional keyword, "
|
||||||
|
"either a :stream (usually tcp), or :datagram (usually udp). The default is :stream.") {
|
||||||
|
janet_arity(argc, 0, 1);
|
||||||
|
|
||||||
|
int socktype = janet_get_sockettype(argv, argc, 0);
|
||||||
|
|
||||||
|
/* Create socket */
|
||||||
|
JSock sfd = JSOCKDEFAULT;
|
||||||
|
struct addrinfo *ai = NULL;
|
||||||
|
struct addrinfo hints;
|
||||||
|
memset(&hints, 0, sizeof(hints));
|
||||||
|
hints.ai_family = AF_UNSPEC;
|
||||||
|
hints.ai_socktype = socktype;
|
||||||
|
hints.ai_flags = 0;
|
||||||
|
int status = getaddrinfo(NULL, "0", &hints, &ai);
|
||||||
|
if (status) {
|
||||||
|
janet_panicf("could not get address info: %s", gai_strerror(status));
|
||||||
|
}
|
||||||
|
|
||||||
|
struct addrinfo *rp = NULL;
|
||||||
|
for (rp = ai; rp != NULL; rp = rp->ai_next) {
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
sfd = WSASocketW(rp->ai_family, rp->ai_socktype | JSOCKFLAGS, rp->ai_protocol, NULL, 0, WSA_FLAG_OVERLAPPED);
|
||||||
|
#else
|
||||||
|
sfd = socket(rp->ai_family, rp->ai_socktype | JSOCKFLAGS, rp->ai_protocol);
|
||||||
|
#endif
|
||||||
|
if (JSOCKVALID(sfd)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
freeaddrinfo(ai);
|
||||||
|
|
||||||
|
if (!JSOCKVALID(sfd)) {
|
||||||
|
Janet v = janet_ev_lasterr();
|
||||||
|
janet_panicf("could not create socket: %V", v);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Wrap socket in abstract type JanetStream */
|
||||||
|
uint32_t udp_flag = 0;
|
||||||
|
if (socktype == SOCK_DGRAM) udp_flag = JANET_STREAM_UDPSERVER;
|
||||||
|
JanetStream *stream = make_stream(sfd, JANET_STREAM_READABLE | JANET_STREAM_WRITABLE | udp_flag);
|
||||||
|
|
||||||
|
/* Set up the socket for non-blocking IO */
|
||||||
|
janet_net_socknoblock(sfd);
|
||||||
|
|
||||||
|
return janet_wrap_abstract(stream);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *serverify_socket(JSock sfd, int reuse_addr, int reuse_port) {
|
||||||
/* Set various socket options */
|
/* Set various socket options */
|
||||||
int enable = 1;
|
int enable = 1;
|
||||||
if (setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, (char *) &enable, sizeof(int)) < 0) {
|
if (reuse_addr) {
|
||||||
return "setsockopt(SO_REUSEADDR) failed";
|
if (setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, (char *) &enable, sizeof(int)) < 0) {
|
||||||
|
return "setsockopt(SO_REUSEADDR) failed";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
if (reuse_port) {
|
||||||
#ifdef SO_REUSEPORT
|
#ifdef SO_REUSEPORT
|
||||||
if (setsockopt(sfd, SOL_SOCKET, SO_REUSEPORT, &enable, sizeof(int)) < 0) {
|
if (setsockopt(sfd, SOL_SOCKET, SO_REUSEPORT, &enable, sizeof(int)) < 0) {
|
||||||
return "setsockopt(SO_REUSEPORT) failed";
|
return "setsockopt(SO_REUSEPORT) failed";
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
(void) reuse_port;
|
||||||
#endif
|
#endif
|
||||||
|
}
|
||||||
janet_net_socknoblock(sfd);
|
janet_net_socknoblock(sfd);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
@@ -642,19 +707,22 @@ JANET_CORE_FN(cfun_net_shutdown,
|
|||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(cfun_net_listen,
|
JANET_CORE_FN(cfun_net_listen,
|
||||||
"(net/listen host port &opt type)",
|
"(net/listen host port &opt type no-reuse)",
|
||||||
"Creates a server. Returns a new stream that is neither readable nor "
|
"Creates a server. Returns a new stream that is neither readable nor "
|
||||||
"writeable. Use net/accept or net/accept-loop be to handle connections and start the server. "
|
"writeable. Use net/accept or net/accept-loop be to handle connections and start the server. "
|
||||||
"The type parameter specifies the type of network connection, either "
|
"The type parameter specifies the type of network connection, either "
|
||||||
"a :stream (usually tcp), or :datagram (usually udp). If not specified, the default is "
|
"a :stream (usually tcp), or :datagram (usually udp). If not specified, the default is "
|
||||||
":stream. The host and port arguments are the same as in net/address.") {
|
":stream. The host and port arguments are the same as in net/address. The last boolean parameter `no-reuse` will "
|
||||||
|
"disable the use of `SO_REUSEADDR` and `SO_REUSEPORT` when creating a server on some operating systems.") {
|
||||||
janet_sandbox_assert(JANET_SANDBOX_NET_LISTEN);
|
janet_sandbox_assert(JANET_SANDBOX_NET_LISTEN);
|
||||||
janet_arity(argc, 2, 3);
|
janet_arity(argc, 2, 4);
|
||||||
|
|
||||||
/* Get host, port, and handler*/
|
/* Get host, port, and handler*/
|
||||||
int socktype = janet_get_sockettype(argv, argc, 2);
|
int socktype = janet_get_sockettype(argv, argc, 2);
|
||||||
int is_unix = 0;
|
int is_unix = 0;
|
||||||
struct addrinfo *ai = janet_get_addrinfo(argv, 0, socktype, 1, &is_unix);
|
socklen_t addrlen = 0;
|
||||||
|
struct addrinfo *ai = janet_get_addrinfo(argv, 0, socktype, 1, &is_unix, &addrlen);
|
||||||
|
int reuse = !(argc >= 4 && janet_truthy(argv[3]));
|
||||||
|
|
||||||
JSock sfd = JSOCKDEFAULT;
|
JSock sfd = JSOCKDEFAULT;
|
||||||
#ifndef JANET_WINDOWS
|
#ifndef JANET_WINDOWS
|
||||||
@@ -664,8 +732,8 @@ JANET_CORE_FN(cfun_net_listen,
|
|||||||
janet_free(ai);
|
janet_free(ai);
|
||||||
janet_panicf("could not create socket: %V", janet_ev_lasterr());
|
janet_panicf("could not create socket: %V", janet_ev_lasterr());
|
||||||
}
|
}
|
||||||
const char *err = serverify_socket(sfd);
|
const char *err = serverify_socket(sfd, reuse, 0);
|
||||||
if (NULL != err || bind(sfd, (struct sockaddr *)ai, sizeof(struct sockaddr_un))) {
|
if (NULL != err || bind(sfd, (struct sockaddr *)ai, addrlen)) {
|
||||||
JSOCKCLOSE(sfd);
|
JSOCKCLOSE(sfd);
|
||||||
janet_free(ai);
|
janet_free(ai);
|
||||||
if (err) {
|
if (err) {
|
||||||
@@ -687,7 +755,7 @@ JANET_CORE_FN(cfun_net_listen,
|
|||||||
sfd = socket(rp->ai_family, rp->ai_socktype | JSOCKFLAGS, rp->ai_protocol);
|
sfd = socket(rp->ai_family, rp->ai_socktype | JSOCKFLAGS, rp->ai_protocol);
|
||||||
#endif
|
#endif
|
||||||
if (!JSOCKVALID(sfd)) continue;
|
if (!JSOCKVALID(sfd)) continue;
|
||||||
const char *err = serverify_socket(sfd);
|
const char *err = serverify_socket(sfd, reuse, reuse);
|
||||||
if (NULL != err) {
|
if (NULL != err) {
|
||||||
JSOCKCLOSE(sfd);
|
JSOCKCLOSE(sfd);
|
||||||
continue;
|
continue;
|
||||||
@@ -1067,6 +1135,7 @@ void janet_lib_net(JanetTable *env) {
|
|||||||
JanetRegExt net_cfuns[] = {
|
JanetRegExt net_cfuns[] = {
|
||||||
JANET_CORE_REG("net/address", cfun_net_sockaddr),
|
JANET_CORE_REG("net/address", cfun_net_sockaddr),
|
||||||
JANET_CORE_REG("net/listen", cfun_net_listen),
|
JANET_CORE_REG("net/listen", cfun_net_listen),
|
||||||
|
JANET_CORE_REG("net/socket", cfun_net_socket),
|
||||||
JANET_CORE_REG("net/accept", cfun_stream_accept),
|
JANET_CORE_REG("net/accept", cfun_stream_accept),
|
||||||
JANET_CORE_REG("net/accept-loop", cfun_stream_accept_loop),
|
JANET_CORE_REG("net/accept-loop", cfun_stream_accept_loop),
|
||||||
JANET_CORE_REG("net/read", cfun_stream_read),
|
JANET_CORE_REG("net/read", cfun_stream_read),
|
||||||
|
|||||||
211
src/core/os.c
211
src/core/os.c
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose and contributors.
|
* Copyright (c) 2025 Calvin Rose and contributors.
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -55,6 +55,7 @@
|
|||||||
#include <sys/utime.h>
|
#include <sys/utime.h>
|
||||||
#include <io.h>
|
#include <io.h>
|
||||||
#include <process.h>
|
#include <process.h>
|
||||||
|
#define JANET_SPAWN_CHDIR
|
||||||
#else
|
#else
|
||||||
#include <spawn.h>
|
#include <spawn.h>
|
||||||
#include <utime.h>
|
#include <utime.h>
|
||||||
@@ -73,6 +74,20 @@ extern char **environ;
|
|||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Detect availability of posix_spawn_file_actions_addchdir_np. Since
|
||||||
|
* this doesn't seem to follow any standard, just a common extension, we
|
||||||
|
* must enumerate supported systems for availability. Define JANET_SPAWN_NO_CHDIR
|
||||||
|
* to disable this. */
|
||||||
|
#ifndef JANET_SPAWN_NO_CHDIR
|
||||||
|
#ifdef __GLIBC__
|
||||||
|
#define JANET_SPAWN_CHDIR
|
||||||
|
#elif defined(JANET_APPLE) /* Some older versions may not work here. */
|
||||||
|
#define JANET_SPAWN_CHDIR
|
||||||
|
#elif defined(__FreeBSD__) /* Not all BSDs work, for example openBSD doesn't seem to support this */
|
||||||
|
#define JANET_SPAWN_CHDIR
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Not POSIX, but all Unixes but Solaris have this function. */
|
/* Not POSIX, but all Unixes but Solaris have this function. */
|
||||||
#if defined(JANET_POSIX) && !defined(__sun)
|
#if defined(JANET_POSIX) && !defined(__sun)
|
||||||
time_t timegm(struct tm *tm);
|
time_t timegm(struct tm *tm);
|
||||||
@@ -158,6 +173,8 @@ JANET_CORE_FN(os_which,
|
|||||||
return janet_ckeywordv("dragonfly");
|
return janet_ckeywordv("dragonfly");
|
||||||
#elif defined(JANET_BSD)
|
#elif defined(JANET_BSD)
|
||||||
return janet_ckeywordv("bsd");
|
return janet_ckeywordv("bsd");
|
||||||
|
#elif defined(JANET_ILLUMOS)
|
||||||
|
return janet_ckeywordv("illumos");
|
||||||
#else
|
#else
|
||||||
return janet_ckeywordv("posix");
|
return janet_ckeywordv("posix");
|
||||||
#endif
|
#endif
|
||||||
@@ -297,6 +314,13 @@ JANET_CORE_FN(os_cpu_count,
|
|||||||
return dflt;
|
return dflt;
|
||||||
}
|
}
|
||||||
return janet_wrap_integer(result);
|
return janet_wrap_integer(result);
|
||||||
|
#elif defined(JANET_ILLUMOS)
|
||||||
|
(void) dflt;
|
||||||
|
long result = sysconf(_SC_NPROCESSORS_CONF);
|
||||||
|
if (result < 0) {
|
||||||
|
return dflt;
|
||||||
|
}
|
||||||
|
return janet_wrap_integer(result);
|
||||||
#else
|
#else
|
||||||
return dflt;
|
return dflt;
|
||||||
#endif
|
#endif
|
||||||
@@ -541,11 +565,12 @@ static void janet_proc_wait_cb(JanetEVGenericMessage args) {
|
|||||||
proc->flags &= ~JANET_PROC_WAITING;
|
proc->flags &= ~JANET_PROC_WAITING;
|
||||||
janet_gcunroot(janet_wrap_abstract(proc));
|
janet_gcunroot(janet_wrap_abstract(proc));
|
||||||
janet_gcunroot(janet_wrap_fiber(args.fiber));
|
janet_gcunroot(janet_wrap_fiber(args.fiber));
|
||||||
if ((status != 0) && (proc->flags & JANET_PROC_ERROR_NONZERO)) {
|
uint32_t sched_id = (uint32_t) args.argi;
|
||||||
JanetString s = janet_formatc("command failed with non-zero exit code %d", status);
|
if (janet_fiber_can_resume(args.fiber) && args.fiber->sched_id == sched_id) {
|
||||||
janet_cancel(args.fiber, janet_wrap_string(s));
|
if ((status != 0) && (proc->flags & JANET_PROC_ERROR_NONZERO)) {
|
||||||
} else {
|
JanetString s = janet_formatc("command failed with non-zero exit code %d", status);
|
||||||
if (janet_fiber_can_resume(args.fiber)) {
|
janet_cancel(args.fiber, janet_wrap_string(s));
|
||||||
|
} else {
|
||||||
janet_schedule(args.fiber, janet_wrap_integer(status));
|
janet_schedule(args.fiber, janet_wrap_integer(status));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -603,6 +628,7 @@ os_proc_wait_impl(JanetProc *proc) {
|
|||||||
memset(&targs, 0, sizeof(targs));
|
memset(&targs, 0, sizeof(targs));
|
||||||
targs.argp = proc;
|
targs.argp = proc;
|
||||||
targs.fiber = janet_root_fiber();
|
targs.fiber = janet_root_fiber();
|
||||||
|
targs.argi = (uint32_t) targs.fiber->sched_id;
|
||||||
janet_gcroot(janet_wrap_abstract(proc));
|
janet_gcroot(janet_wrap_abstract(proc));
|
||||||
janet_gcroot(janet_wrap_fiber(targs.fiber));
|
janet_gcroot(janet_wrap_fiber(targs.fiber));
|
||||||
janet_ev_threaded_call(janet_proc_wait_subr, targs, janet_proc_wait_cb);
|
janet_ev_threaded_call(janet_proc_wait_subr, targs, janet_proc_wait_cb);
|
||||||
@@ -629,16 +655,15 @@ os_proc_wait_impl(JanetProc *proc) {
|
|||||||
|
|
||||||
JANET_CORE_FN(os_proc_wait,
|
JANET_CORE_FN(os_proc_wait,
|
||||||
"(os/proc-wait proc)",
|
"(os/proc-wait proc)",
|
||||||
"Suspend the current fiber until the subprocess completes. Returns the subprocess return code. "
|
"Suspend the current fiber until the subprocess `proc` completes. Once `proc` "
|
||||||
"os/proc-wait cannot be called twice on the same process. If `ev/with-deadline` cancels `os/proc-wait` "
|
"completes, return the exit code of `proc`. If called more than once on the same "
|
||||||
"with an error or os/proc-wait is cancelled with any error caused by anything else, os/proc-wait still "
|
"core/process value, will raise an error. When creating subprocesses using "
|
||||||
"finishes in the background. Only after os/proc-wait finishes, a process is cleaned up by the operating "
|
"`os/spawn`, this function should be called on the returned value to avoid zombie "
|
||||||
"system. Thus, a process becomes a zombie process if os/proc-wait is not called.") {
|
"processes.") {
|
||||||
janet_fixarity(argc, 1);
|
janet_fixarity(argc, 1);
|
||||||
JanetProc *proc = janet_getabstract(argv, 0, &ProcAT);
|
JanetProc *proc = janet_getabstract(argv, 0, &ProcAT);
|
||||||
#ifdef JANET_EV
|
#ifdef JANET_EV
|
||||||
os_proc_wait_impl(proc);
|
os_proc_wait_impl(proc);
|
||||||
return janet_wrap_nil();
|
|
||||||
#else
|
#else
|
||||||
return os_proc_wait_impl(proc);
|
return os_proc_wait_impl(proc);
|
||||||
#endif
|
#endif
|
||||||
@@ -743,12 +768,13 @@ static int get_signal_kw(const Janet *argv, int32_t n) {
|
|||||||
|
|
||||||
JANET_CORE_FN(os_proc_kill,
|
JANET_CORE_FN(os_proc_kill,
|
||||||
"(os/proc-kill proc &opt wait signal)",
|
"(os/proc-kill proc &opt wait signal)",
|
||||||
"Kill a subprocess by sending SIGKILL to it on posix systems, or by closing the process "
|
"Kill the subprocess `proc` by sending SIGKILL to it on POSIX systems, or by closing "
|
||||||
"handle on windows. If os/proc-wait already finished for proc, os/proc-kill raises an error. After "
|
"the process handle on Windows. If `proc` has already completed, raise an error. If "
|
||||||
"sending signal to proc, if `wait` is truthy, will wait for the process to finish and return the exit "
|
"`wait` is truthy, will wait for `proc` to complete and return the exit code (this "
|
||||||
"code by calling os/proc-wait. Otherwise, returns `proc`. If signal is specified, send it instead. "
|
"will raise an error if `proc` is being waited for). Otherwise, return `proc`. If "
|
||||||
"Signal keywords are named after their C counterparts but in lowercase with the leading `SIG` stripped. "
|
"`signal` is provided, send it instead of SIGKILL. Signal keywords are named after "
|
||||||
"Signals are ignored on windows.") {
|
"their C counterparts but in lowercase with the leading SIG stripped. `signal` is "
|
||||||
|
"ignored on Windows.") {
|
||||||
janet_arity(argc, 1, 3);
|
janet_arity(argc, 1, 3);
|
||||||
JanetProc *proc = janet_getabstract(argv, 0, &ProcAT);
|
JanetProc *proc = janet_getabstract(argv, 0, &ProcAT);
|
||||||
if (proc->flags & JANET_PROC_WAITED) {
|
if (proc->flags & JANET_PROC_WAITED) {
|
||||||
@@ -776,7 +802,6 @@ JANET_CORE_FN(os_proc_kill,
|
|||||||
if (argc > 1 && janet_truthy(argv[1])) {
|
if (argc > 1 && janet_truthy(argv[1])) {
|
||||||
#ifdef JANET_EV
|
#ifdef JANET_EV
|
||||||
os_proc_wait_impl(proc);
|
os_proc_wait_impl(proc);
|
||||||
return janet_wrap_nil();
|
|
||||||
#else
|
#else
|
||||||
return os_proc_wait_impl(proc);
|
return os_proc_wait_impl(proc);
|
||||||
#endif
|
#endif
|
||||||
@@ -787,9 +812,9 @@ JANET_CORE_FN(os_proc_kill,
|
|||||||
|
|
||||||
JANET_CORE_FN(os_proc_close,
|
JANET_CORE_FN(os_proc_close,
|
||||||
"(os/proc-close proc)",
|
"(os/proc-close proc)",
|
||||||
"Close pipes created by `os/spawn` if they have not been closed. Then, if os/proc-wait was not already "
|
"Close pipes created for subprocess `proc` by `os/spawn` if they have not been "
|
||||||
"called on proc, os/proc-wait is called on it, and it returns the exit code returned by os/proc-wait. "
|
"closed. Then, if `proc` is not being waited for, wait. If this function waits, when "
|
||||||
"Otherwise, returns nil.") {
|
"`proc` completes, return the exit code of `proc`. Otherwise, return nil.") {
|
||||||
janet_fixarity(argc, 1);
|
janet_fixarity(argc, 1);
|
||||||
JanetProc *proc = janet_getabstract(argv, 0, &ProcAT);
|
JanetProc *proc = janet_getabstract(argv, 0, &ProcAT);
|
||||||
#ifdef JANET_EV
|
#ifdef JANET_EV
|
||||||
@@ -807,12 +832,24 @@ JANET_CORE_FN(os_proc_close,
|
|||||||
}
|
}
|
||||||
#ifdef JANET_EV
|
#ifdef JANET_EV
|
||||||
os_proc_wait_impl(proc);
|
os_proc_wait_impl(proc);
|
||||||
return janet_wrap_nil();
|
|
||||||
#else
|
#else
|
||||||
return os_proc_wait_impl(proc);
|
return os_proc_wait_impl(proc);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
JANET_CORE_FN(os_proc_getpid,
|
||||||
|
"(os/getpid)",
|
||||||
|
"Get the process ID of the current process.") {
|
||||||
|
janet_sandbox_assert(JANET_SANDBOX_SUBPROCESS);
|
||||||
|
janet_fixarity(argc, 0);
|
||||||
|
(void) argv;
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
return janet_wrap_number((double) _getpid());
|
||||||
|
#else
|
||||||
|
return janet_wrap_number((double) getpid());
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
static void swap_handles(JanetHandle *handles) {
|
static void swap_handles(JanetHandle *handles) {
|
||||||
JanetHandle temp = handles[0];
|
JanetHandle temp = handles[0];
|
||||||
handles[0] = handles[1];
|
handles[0] = handles[1];
|
||||||
@@ -1137,6 +1174,7 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
JanetAbstract orig_in = NULL, orig_out = NULL, orig_err = NULL;
|
JanetAbstract orig_in = NULL, orig_out = NULL, orig_err = NULL;
|
||||||
JanetHandle new_in = JANET_HANDLE_NONE, new_out = JANET_HANDLE_NONE, new_err = JANET_HANDLE_NONE;
|
JanetHandle new_in = JANET_HANDLE_NONE, new_out = JANET_HANDLE_NONE, new_err = JANET_HANDLE_NONE;
|
||||||
JanetHandle pipe_in = JANET_HANDLE_NONE, pipe_out = JANET_HANDLE_NONE, pipe_err = JANET_HANDLE_NONE;
|
JanetHandle pipe_in = JANET_HANDLE_NONE, pipe_out = JANET_HANDLE_NONE, pipe_err = JANET_HANDLE_NONE;
|
||||||
|
int stderr_is_stdout = 0;
|
||||||
int pipe_errflag = 0; /* Track errors setting up pipes */
|
int pipe_errflag = 0; /* Track errors setting up pipes */
|
||||||
int pipe_owner_flags = (is_spawn && (flags & 0x8)) ? JANET_PROC_ALLOW_ZOMBIE : 0;
|
int pipe_owner_flags = (is_spawn && (flags & 0x8)) ? JANET_PROC_ALLOW_ZOMBIE : 0;
|
||||||
|
|
||||||
@@ -1161,11 +1199,28 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
if (is_spawn && janet_keyeq(maybe_stderr, "pipe")) {
|
if (is_spawn && janet_keyeq(maybe_stderr, "pipe")) {
|
||||||
new_err = make_pipes(&pipe_err, 0, &pipe_errflag);
|
new_err = make_pipes(&pipe_err, 0, &pipe_errflag);
|
||||||
pipe_owner_flags |= JANET_PROC_OWNS_STDERR;
|
pipe_owner_flags |= JANET_PROC_OWNS_STDERR;
|
||||||
|
} else if (is_spawn && janet_keyeq(maybe_stderr, "out")) {
|
||||||
|
stderr_is_stdout = 1;
|
||||||
} else if (!janet_checktype(maybe_stderr, JANET_NIL)) {
|
} else if (!janet_checktype(maybe_stderr, JANET_NIL)) {
|
||||||
new_err = janet_getjstream(&maybe_stderr, 0, &orig_err);
|
new_err = janet_getjstream(&maybe_stderr, 0, &orig_err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Optional working directory. Available for both os/execute and os/spawn. */
|
||||||
|
const char *chdir_path = NULL;
|
||||||
|
if (argc > 2) {
|
||||||
|
JanetDictView tab = janet_getdictionary(argv, 2);
|
||||||
|
Janet workdir = janet_dictionary_get(tab.kvs, tab.cap, janet_ckeywordv("cd"));
|
||||||
|
if (janet_checktype(workdir, JANET_STRING)) {
|
||||||
|
chdir_path = (const char *) janet_unwrap_string(workdir);
|
||||||
|
#ifndef JANET_SPAWN_CHDIR
|
||||||
|
janet_panicf(":cd argument not supported on this system - %s", chdir_path);
|
||||||
|
#endif
|
||||||
|
} else if (!janet_checktype(workdir, JANET_NIL)) {
|
||||||
|
janet_panicf("expected string for :cd argumnet, got %v", workdir);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Clean up if any of the pipes have any issues */
|
/* Clean up if any of the pipes have any issues */
|
||||||
if (pipe_errflag) {
|
if (pipe_errflag) {
|
||||||
if (pipe_in != JANET_HANDLE_NONE) close_handle(pipe_in);
|
if (pipe_in != JANET_HANDLE_NONE) close_handle(pipe_in);
|
||||||
@@ -1180,6 +1235,7 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
SECURITY_ATTRIBUTES saAttr;
|
SECURITY_ATTRIBUTES saAttr;
|
||||||
PROCESS_INFORMATION processInfo;
|
PROCESS_INFORMATION processInfo;
|
||||||
STARTUPINFO startupInfo;
|
STARTUPINFO startupInfo;
|
||||||
|
LPCSTR lpCurrentDirectory = NULL;
|
||||||
memset(&saAttr, 0, sizeof(saAttr));
|
memset(&saAttr, 0, sizeof(saAttr));
|
||||||
memset(&processInfo, 0, sizeof(processInfo));
|
memset(&processInfo, 0, sizeof(processInfo));
|
||||||
memset(&startupInfo, 0, sizeof(startupInfo));
|
memset(&startupInfo, 0, sizeof(startupInfo));
|
||||||
@@ -1196,6 +1252,10 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
}
|
}
|
||||||
const char *path = (const char *) janet_unwrap_string(exargs.items[0]);
|
const char *path = (const char *) janet_unwrap_string(exargs.items[0]);
|
||||||
|
|
||||||
|
if (chdir_path != NULL) {
|
||||||
|
lpCurrentDirectory = chdir_path;
|
||||||
|
}
|
||||||
|
|
||||||
/* Do IO redirection */
|
/* Do IO redirection */
|
||||||
|
|
||||||
if (pipe_in != JANET_HANDLE_NONE) {
|
if (pipe_in != JANET_HANDLE_NONE) {
|
||||||
@@ -1203,7 +1263,7 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
} else if (new_in != JANET_HANDLE_NONE) {
|
} else if (new_in != JANET_HANDLE_NONE) {
|
||||||
startupInfo.hStdInput = new_in;
|
startupInfo.hStdInput = new_in;
|
||||||
} else {
|
} else {
|
||||||
startupInfo.hStdInput = (HANDLE) _get_osfhandle(0);
|
startupInfo.hStdInput = (HANDLE) _get_osfhandle(_fileno(stdin));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pipe_out != JANET_HANDLE_NONE) {
|
if (pipe_out != JANET_HANDLE_NONE) {
|
||||||
@@ -1211,15 +1271,17 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
} else if (new_out != JANET_HANDLE_NONE) {
|
} else if (new_out != JANET_HANDLE_NONE) {
|
||||||
startupInfo.hStdOutput = new_out;
|
startupInfo.hStdOutput = new_out;
|
||||||
} else {
|
} else {
|
||||||
startupInfo.hStdOutput = (HANDLE) _get_osfhandle(1);
|
startupInfo.hStdOutput = (HANDLE) _get_osfhandle(_fileno(stdout));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pipe_err != JANET_HANDLE_NONE) {
|
if (pipe_err != JANET_HANDLE_NONE) {
|
||||||
startupInfo.hStdError = pipe_err;
|
startupInfo.hStdError = pipe_err;
|
||||||
} else if (new_err != NULL) {
|
} else if (new_err != NULL) {
|
||||||
startupInfo.hStdError = new_err;
|
startupInfo.hStdError = new_err;
|
||||||
|
} else if (stderr_is_stdout) {
|
||||||
|
startupInfo.hStdError = startupInfo.hStdOutput;
|
||||||
} else {
|
} else {
|
||||||
startupInfo.hStdError = (HANDLE) _get_osfhandle(2);
|
startupInfo.hStdError = (HANDLE) _get_osfhandle(_fileno(stderr));
|
||||||
}
|
}
|
||||||
|
|
||||||
int cp_failed = 0;
|
int cp_failed = 0;
|
||||||
@@ -1230,7 +1292,7 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
TRUE, /* handle inheritance */
|
TRUE, /* handle inheritance */
|
||||||
0, /* flags */
|
0, /* flags */
|
||||||
use_environ ? NULL : envp, /* pass in environment */
|
use_environ ? NULL : envp, /* pass in environment */
|
||||||
NULL, /* use parents starting directory */
|
lpCurrentDirectory,
|
||||||
&startupInfo,
|
&startupInfo,
|
||||||
&processInfo)) {
|
&processInfo)) {
|
||||||
cp_failed = 1;
|
cp_failed = 1;
|
||||||
@@ -1268,9 +1330,6 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
|
|
||||||
/* exec mode */
|
/* exec mode */
|
||||||
if (mode == JANET_EXECUTE_EXEC) {
|
if (mode == JANET_EXECUTE_EXEC) {
|
||||||
#ifdef JANET_WINDOWS
|
|
||||||
janet_panic("not supported on windows");
|
|
||||||
#else
|
|
||||||
int status;
|
int status;
|
||||||
if (!use_environ) {
|
if (!use_environ) {
|
||||||
environ = envp;
|
environ = envp;
|
||||||
@@ -1283,7 +1342,6 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
}
|
}
|
||||||
} while (status == -1 && errno == EINTR);
|
} while (status == -1 && errno == EINTR);
|
||||||
janet_panicf("%p: %s", cargv[0], janet_strerror(errno ? errno : ENOENT));
|
janet_panicf("%p: %s", cargv[0], janet_strerror(errno ? errno : ENOENT));
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Use posix_spawn to spawn new process */
|
/* Use posix_spawn to spawn new process */
|
||||||
@@ -1291,6 +1349,15 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
/* Posix spawn setup */
|
/* Posix spawn setup */
|
||||||
posix_spawn_file_actions_t actions;
|
posix_spawn_file_actions_t actions;
|
||||||
posix_spawn_file_actions_init(&actions);
|
posix_spawn_file_actions_init(&actions);
|
||||||
|
#ifdef JANET_SPAWN_CHDIR
|
||||||
|
if (chdir_path != NULL) {
|
||||||
|
#ifdef JANET_SPAWN_CHDIR_NO_NP
|
||||||
|
posix_spawn_file_actions_addchdir(&actions, chdir_path);
|
||||||
|
#else
|
||||||
|
posix_spawn_file_actions_addchdir_np(&actions, chdir_path);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
#endif
|
||||||
if (pipe_in != JANET_HANDLE_NONE) {
|
if (pipe_in != JANET_HANDLE_NONE) {
|
||||||
posix_spawn_file_actions_adddup2(&actions, pipe_in, 0);
|
posix_spawn_file_actions_adddup2(&actions, pipe_in, 0);
|
||||||
posix_spawn_file_actions_addclose(&actions, pipe_in);
|
posix_spawn_file_actions_addclose(&actions, pipe_in);
|
||||||
@@ -1313,6 +1380,8 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
} else if (new_err != JANET_HANDLE_NONE && new_err != 2) {
|
} else if (new_err != JANET_HANDLE_NONE && new_err != 2) {
|
||||||
posix_spawn_file_actions_adddup2(&actions, new_err, 2);
|
posix_spawn_file_actions_adddup2(&actions, new_err, 2);
|
||||||
posix_spawn_file_actions_addclose(&actions, new_err);
|
posix_spawn_file_actions_addclose(&actions, new_err);
|
||||||
|
} else if (stderr_is_stdout) {
|
||||||
|
posix_spawn_file_actions_adddup2(&actions, 1, 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
pid_t pid;
|
pid_t pid;
|
||||||
@@ -1384,45 +1453,57 @@ static Janet os_execute_impl(int32_t argc, Janet *argv, JanetExecuteMode mode) {
|
|||||||
|
|
||||||
JANET_CORE_FN(os_execute,
|
JANET_CORE_FN(os_execute,
|
||||||
"(os/execute args &opt flags env)",
|
"(os/execute args &opt flags env)",
|
||||||
"Execute a program on the system and pass it string arguments. `flags` "
|
"Execute a program on the system and return the exit code. `args` is an array/tuple "
|
||||||
"is a keyword that modifies how the program will execute.\n"
|
"of strings. The first string is the name of the program and the remainder are "
|
||||||
"* :e - enables passing an environment to the program. Without :e, the "
|
"arguments passed to the program. `flags` is a keyword made from the following "
|
||||||
|
"characters that modifies how the program executes:\n"
|
||||||
|
"* :e - enables passing an environment to the program. Without 'e', the "
|
||||||
"current environment is inherited.\n"
|
"current environment is inherited.\n"
|
||||||
"* :p - allows searching the current PATH for the binary to execute. "
|
"* :p - allows searching the current PATH for the program to execute. "
|
||||||
"Without this flag, binaries must use absolute paths.\n"
|
"Without this flag, the first element of `args` must be an absolute path.\n"
|
||||||
"* :x - raise error if exit code is non-zero.\n"
|
"* :x - raises error if exit code is non-zero.\n"
|
||||||
"* :d - Don't try and terminate the process on garbage collection (allow spawning zombies).\n"
|
"* :d - prevents the garbage collector terminating the program (if still running) "
|
||||||
"`env` is a table or struct mapping environment variables to values. It can also "
|
"and calling the equivalent of `os/proc-wait` (allows zombie processes).\n"
|
||||||
"contain the keys :in, :out, and :err, which allow redirecting stdio in the subprocess. "
|
"`env` is a table/struct mapping environment variables to values. It can also "
|
||||||
":in, :out, and :err should be core/file values or core/stream values. core/file values and core/stream "
|
"contain the keys :in, :out, and :err, which allow redirecting stdio in the "
|
||||||
"values passed to :in, :out, and :err should be closed manually because os/execute doesn't close them. "
|
"subprocess. :in, :out, and :err should be core/file or core/stream values. "
|
||||||
"Returns the exit code of the program.") {
|
"If core/stream values are used, the caller is responsible for ensuring pipes do not "
|
||||||
|
"cause the program to block and deadlock.") {
|
||||||
return os_execute_impl(argc, argv, JANET_EXECUTE_EXECUTE);
|
return os_execute_impl(argc, argv, JANET_EXECUTE_EXECUTE);
|
||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(os_spawn,
|
JANET_CORE_FN(os_spawn,
|
||||||
"(os/spawn args &opt flags env)",
|
"(os/spawn args &opt flags env)",
|
||||||
"Execute a program on the system and return a handle to the process. Otherwise, takes the "
|
"Execute a program on the system and return a core/process value representing the "
|
||||||
"same arguments as `os/execute`. Does not wait for the process. For each of the :in, :out, and :err keys "
|
"spawned subprocess. Takes the same arguments as `os/execute` but does not wait for "
|
||||||
"of the `env` argument, one can also pass in the keyword `:pipe` to get streams for standard IO of the "
|
"the subprocess to complete. Unlike `os/execute`, the value `:pipe` can be used for "
|
||||||
"subprocess that can be read from and written to. The returned value `proc` has the fields :in, :out, "
|
":in, :out and :err keys in `env`. If used, the returned core/process will have a "
|
||||||
":err, and the additional field :pid on unix-like platforms. `(os/proc-wait proc)` must be called to "
|
"writable stream in the :in field and readable streams in the :out and :err fields. "
|
||||||
"rejoin the subprocess. After `(os/proc-wait proc)` finishes, proc gains a new field, :return-code. "
|
"On non-Windows systems, the subprocess PID will be in the :pid field. The caller is "
|
||||||
"If :x flag is passed to os/spawn, non-zero exit code will cause os/proc-wait to raise an error. "
|
"responsible for waiting on the process (e.g. by calling `os/proc-wait` on the "
|
||||||
"If pipe streams created with :pipe keyword are not closed in time, janet can run out of file "
|
"returned core/process value) to avoid creating zombie process. After the subprocess "
|
||||||
"descriptors. They can be closed individually, or `os/proc-close` can close all pipe streams on proc. "
|
"completes, the exit value is in the :return-code field. If `flags` includes 'x', a "
|
||||||
"If pipe streams aren't read before `os/proc-wait` finishes, then pipe buffers become full, and the "
|
"non-zero exit code will cause a waiting fiber to raise an error. The use of "
|
||||||
"process cannot finish because the process cannot print more on pipe buffers which are already full. "
|
"`:pipe` may fail if there are too many active file descriptors. The caller is "
|
||||||
"If the process cannot finish, os/proc-wait cannot finish, either.") {
|
"responsible for closing pipes created by `:pipe` (either individually or using "
|
||||||
|
"`os/proc-close`). Similar to `os/execute`, the caller is responsible for ensuring "
|
||||||
|
"pipes do not cause the program to block and deadlock. As a special case, the stream passed to `:err` "
|
||||||
|
"can be the keyword `:out` to redirect stderr to stdout in the subprocess.") {
|
||||||
return os_execute_impl(argc, argv, JANET_EXECUTE_SPAWN);
|
return os_execute_impl(argc, argv, JANET_EXECUTE_SPAWN);
|
||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(os_posix_exec,
|
JANET_CORE_FN(os_posix_exec,
|
||||||
"(os/posix-exec args &opt flags env)",
|
"(os/posix-exec args &opt flags env)",
|
||||||
"Use the execvpe or execve system calls to replace the current process with an interface similar to os/execute. "
|
"Use the execvpe or execve system calls to replace the current process with an interface similar to os/execute. "
|
||||||
"However, instead of creating a subprocess, the current process is replaced. Is not supported on windows, and "
|
"However, instead of creating a subprocess, the current process is replaced. Is not supported on Windows, and "
|
||||||
"does not allow redirection of stdio.") {
|
"does not allow redirection of stdio.") {
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
(void) argc;
|
||||||
|
(void) argv;
|
||||||
|
janet_panic("not supported on Windows");
|
||||||
|
#else
|
||||||
return os_execute_impl(argc, argv, JANET_EXECUTE_EXEC);
|
return os_execute_impl(argc, argv, JANET_EXECUTE_EXEC);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(os_posix_fork,
|
JANET_CORE_FN(os_posix_fork,
|
||||||
@@ -1433,7 +1514,7 @@ JANET_CORE_FN(os_posix_fork,
|
|||||||
janet_fixarity(argc, 0);
|
janet_fixarity(argc, 0);
|
||||||
(void) argv;
|
(void) argv;
|
||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
janet_panic("not supported");
|
janet_panic("not supported on Windows");
|
||||||
#else
|
#else
|
||||||
pid_t result;
|
pid_t result;
|
||||||
do {
|
do {
|
||||||
@@ -1880,7 +1961,6 @@ JANET_CORE_FN(os_mktime,
|
|||||||
/* utc time */
|
/* utc time */
|
||||||
#ifdef JANET_NO_UTC_MKTIME
|
#ifdef JANET_NO_UTC_MKTIME
|
||||||
janet_panic("os/mktime UTC not supported on this platform");
|
janet_panic("os/mktime UTC not supported on this platform");
|
||||||
return janet_wrap_nil();
|
|
||||||
#else
|
#else
|
||||||
t = timegm(&t_info);
|
t = timegm(&t_info);
|
||||||
#endif
|
#endif
|
||||||
@@ -1947,8 +2027,7 @@ JANET_CORE_FN(os_link,
|
|||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
(void) argc;
|
(void) argc;
|
||||||
(void) argv;
|
(void) argv;
|
||||||
janet_panic("os/link not supported on Windows");
|
janet_panic("not supported on Windows");
|
||||||
return janet_wrap_nil();
|
|
||||||
#else
|
#else
|
||||||
const char *oldpath = janet_getcstring(argv, 0);
|
const char *oldpath = janet_getcstring(argv, 0);
|
||||||
const char *newpath = janet_getcstring(argv, 1);
|
const char *newpath = janet_getcstring(argv, 1);
|
||||||
@@ -1966,8 +2045,7 @@ JANET_CORE_FN(os_symlink,
|
|||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
(void) argc;
|
(void) argc;
|
||||||
(void) argv;
|
(void) argv;
|
||||||
janet_panic("os/symlink not supported on Windows");
|
janet_panic("not supported on Windows");
|
||||||
return janet_wrap_nil();
|
|
||||||
#else
|
#else
|
||||||
const char *oldpath = janet_getcstring(argv, 0);
|
const char *oldpath = janet_getcstring(argv, 0);
|
||||||
const char *newpath = janet_getcstring(argv, 1);
|
const char *newpath = janet_getcstring(argv, 1);
|
||||||
@@ -2069,8 +2147,7 @@ JANET_CORE_FN(os_readlink,
|
|||||||
#ifdef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
(void) argc;
|
(void) argc;
|
||||||
(void) argv;
|
(void) argv;
|
||||||
janet_panic("os/readlink not supported on Windows");
|
janet_panic("not supported on Windows");
|
||||||
return janet_wrap_nil();
|
|
||||||
#else
|
#else
|
||||||
static char buffer[PATH_MAX];
|
static char buffer[PATH_MAX];
|
||||||
const char *path = janet_getcstring(argv, 0);
|
const char *path = janet_getcstring(argv, 0);
|
||||||
@@ -2326,7 +2403,6 @@ static Janet os_stat_or_lstat(int do_lstat, int32_t argc, Janet *argv) {
|
|||||||
return sg->fn(&st);
|
return sg->fn(&st);
|
||||||
}
|
}
|
||||||
janet_panicf("unexpected keyword %v", janet_wrap_keyword(key));
|
janet_panicf("unexpected keyword %v", janet_wrap_keyword(key));
|
||||||
return janet_wrap_nil();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2795,6 +2871,10 @@ void janet_lib_os(JanetTable *env) {
|
|||||||
JANET_CORE_REG("os/proc-wait", os_proc_wait),
|
JANET_CORE_REG("os/proc-wait", os_proc_wait),
|
||||||
JANET_CORE_REG("os/proc-kill", os_proc_kill),
|
JANET_CORE_REG("os/proc-kill", os_proc_kill),
|
||||||
JANET_CORE_REG("os/proc-close", os_proc_close),
|
JANET_CORE_REG("os/proc-close", os_proc_close),
|
||||||
|
JANET_CORE_REG("os/getpid", os_proc_getpid),
|
||||||
|
#ifdef JANET_EV
|
||||||
|
JANET_CORE_REG("os/sigaction", os_sigaction),
|
||||||
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* high resolution timers */
|
/* high resolution timers */
|
||||||
@@ -2803,7 +2883,6 @@ void janet_lib_os(JanetTable *env) {
|
|||||||
#ifdef JANET_EV
|
#ifdef JANET_EV
|
||||||
JANET_CORE_REG("os/open", os_open), /* fs read and write */
|
JANET_CORE_REG("os/open", os_open), /* fs read and write */
|
||||||
JANET_CORE_REG("os/pipe", os_pipe),
|
JANET_CORE_REG("os/pipe", os_pipe),
|
||||||
JANET_CORE_REG("os/sigaction", os_sigaction),
|
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
JANET_REG_END
|
JANET_REG_END
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -363,8 +363,7 @@ static int stringend(JanetParser *p, JanetParseState *state) {
|
|||||||
JanetParseState top = p->states[p->statecount - 1];
|
JanetParseState top = p->states[p->statecount - 1];
|
||||||
int32_t indent_col = (int32_t) top.column - 1;
|
int32_t indent_col = (int32_t) top.column - 1;
|
||||||
uint8_t *r = bufstart, *end = r + buflen;
|
uint8_t *r = bufstart, *end = r + buflen;
|
||||||
/* Check if there are any characters before the start column -
|
/* Unless there are only spaces before EOLs, disable reindenting */
|
||||||
* if so, do not reindent. */
|
|
||||||
int reindent = 1;
|
int reindent = 1;
|
||||||
while (reindent && (r < end)) {
|
while (reindent && (r < end)) {
|
||||||
if (*r++ == '\n') {
|
if (*r++ == '\n') {
|
||||||
@@ -374,34 +373,36 @@ static int stringend(JanetParser *p, JanetParseState *state) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if ((r + 1) < end && *r == '\r' && *(r + 1) == '\n') reindent = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* Now reindent if able to, otherwise just drop leading newline. */
|
/* Now reindent if able */
|
||||||
if (!reindent) {
|
if (reindent) {
|
||||||
if (buflen > 0 && bufstart[0] == '\n') {
|
|
||||||
buflen--;
|
|
||||||
bufstart++;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
uint8_t *w = bufstart;
|
uint8_t *w = bufstart;
|
||||||
r = bufstart;
|
r = bufstart;
|
||||||
while (r < end) {
|
while (r < end) {
|
||||||
if (*r == '\n') {
|
if (*r == '\n') {
|
||||||
if (r == bufstart) {
|
*w++ = *r++;
|
||||||
/* Skip leading newline */
|
|
||||||
r++;
|
|
||||||
} else {
|
|
||||||
*w++ = *r++;
|
|
||||||
}
|
|
||||||
for (int32_t j = 0; (r < end) && (*r != '\n') && (j < indent_col); j++, r++);
|
for (int32_t j = 0; (r < end) && (*r != '\n') && (j < indent_col); j++, r++);
|
||||||
|
if ((r + 1) < end && *r == '\r' && *(r + 1) == '\n') *w++ = *r++;
|
||||||
} else {
|
} else {
|
||||||
*w++ = *r++;
|
*w++ = *r++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
buflen = (int32_t)(w - bufstart);
|
buflen = (int32_t)(w - bufstart);
|
||||||
}
|
}
|
||||||
/* Check for trailing newline character so we can remove it */
|
/* Check for leading EOL so we can remove it */
|
||||||
if (buflen > 0 && bufstart[buflen - 1] == '\n') {
|
if (buflen > 1 && bufstart[0] == '\r' && bufstart[1] == '\n') { /* Windows EOL */
|
||||||
|
buflen = buflen - 2;
|
||||||
|
bufstart = bufstart + 2;
|
||||||
|
} else if (buflen > 0 && bufstart[0] == '\n') { /* Unix EOL */
|
||||||
|
buflen--;
|
||||||
|
bufstart++;
|
||||||
|
}
|
||||||
|
/* Check for trailing EOL so we can remove it */
|
||||||
|
if (buflen > 1 && bufstart[buflen - 2] == '\r' && bufstart[buflen - 1] == '\n') { /* Windows EOL */
|
||||||
|
buflen = buflen - 2;
|
||||||
|
} else if (buflen > 0 && bufstart[buflen - 1] == '\n') { /* Unix EOL */
|
||||||
buflen--;
|
buflen--;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -342,7 +342,7 @@ tail:
|
|||||||
while (captured < hi) {
|
while (captured < hi) {
|
||||||
CapState cs2 = cap_save(s);
|
CapState cs2 = cap_save(s);
|
||||||
next_text = peg_rule(s, rule_a, text);
|
next_text = peg_rule(s, rule_a, text);
|
||||||
if (!next_text || next_text == text) {
|
if (!next_text || ((next_text == text) && (hi == UINT32_MAX))) {
|
||||||
cap_load(s, cs2);
|
cap_load(s, cs2);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -544,41 +544,80 @@ tail:
|
|||||||
return window_end;
|
return window_end;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case RULE_TIL: {
|
||||||
|
const uint32_t *rule_terminus = s->bytecode + rule[1];
|
||||||
|
const uint32_t *rule_subpattern = s->bytecode + rule[2];
|
||||||
|
|
||||||
|
const uint8_t *terminus_start = text;
|
||||||
|
const uint8_t *terminus_end = NULL;
|
||||||
|
down1(s);
|
||||||
|
while (terminus_start <= s->text_end) {
|
||||||
|
CapState cs2 = cap_save(s);
|
||||||
|
terminus_end = peg_rule(s, rule_terminus, terminus_start);
|
||||||
|
cap_load(s, cs2);
|
||||||
|
if (terminus_end) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
terminus_start++;
|
||||||
|
}
|
||||||
|
up1(s);
|
||||||
|
|
||||||
|
if (!terminus_end) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
const uint8_t *saved_end = s->text_end;
|
||||||
|
s->text_end = terminus_start;
|
||||||
|
down1(s);
|
||||||
|
const uint8_t *matched = peg_rule(s, rule_subpattern, text);
|
||||||
|
up1(s);
|
||||||
|
s->text_end = saved_end;
|
||||||
|
|
||||||
|
if (!matched) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return terminus_end;
|
||||||
|
}
|
||||||
|
|
||||||
case RULE_SPLIT: {
|
case RULE_SPLIT: {
|
||||||
const uint8_t *saved_end = s->text_end;
|
const uint8_t *saved_end = s->text_end;
|
||||||
const uint32_t *rule_separator = s->bytecode + rule[1];
|
const uint32_t *rule_separator = s->bytecode + rule[1];
|
||||||
const uint32_t *rule_subpattern = s->bytecode + rule[2];
|
const uint32_t *rule_subpattern = s->bytecode + rule[2];
|
||||||
|
|
||||||
const uint8_t *separator_end = NULL;
|
const uint8_t *chunk_start = text;
|
||||||
do {
|
const uint8_t *chunk_end = NULL;
|
||||||
const uint8_t *text_start = text;
|
|
||||||
|
while (text <= saved_end) {
|
||||||
|
/* Find next split (or end of text) */
|
||||||
CapState cs = cap_save(s);
|
CapState cs = cap_save(s);
|
||||||
down1(s);
|
down1(s);
|
||||||
while (text <= s->text_end) {
|
while (text <= saved_end) {
|
||||||
separator_end = peg_rule(s, rule_separator, text);
|
chunk_end = text;
|
||||||
|
const uint8_t *check = peg_rule(s, rule_separator, text);
|
||||||
cap_load(s, cs);
|
cap_load(s, cs);
|
||||||
if (separator_end) {
|
if (check) {
|
||||||
|
text = check;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
text++;
|
text++;
|
||||||
}
|
}
|
||||||
up1(s);
|
up1(s);
|
||||||
|
|
||||||
if (separator_end) {
|
/* Match between splits */
|
||||||
s->text_end = text;
|
s->text_end = chunk_end;
|
||||||
text = separator_end;
|
|
||||||
}
|
|
||||||
|
|
||||||
down1(s);
|
down1(s);
|
||||||
const uint8_t *subpattern_end = peg_rule(s, rule_subpattern, text_start);
|
const uint8_t *subpattern_end = peg_rule(s, rule_subpattern, chunk_start);
|
||||||
up1(s);
|
up1(s);
|
||||||
s->text_end = saved_end;
|
s->text_end = saved_end;
|
||||||
|
if (!subpattern_end) return NULL; /* Don't match anything */
|
||||||
|
|
||||||
if (!subpattern_end) {
|
/* Ensure forward progress */
|
||||||
return NULL;
|
if (text == chunk_start) return NULL;
|
||||||
}
|
chunk_start = text;
|
||||||
} while (separator_end);
|
}
|
||||||
|
|
||||||
|
s->text_end = saved_end;
|
||||||
return s->text_end;
|
return s->text_end;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1227,6 +1266,14 @@ static void spec_sub(Builder *b, int32_t argc, const Janet *argv) {
|
|||||||
emit_2(r, RULE_SUB, subrule1, subrule2);
|
emit_2(r, RULE_SUB, subrule1, subrule2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void spec_til(Builder *b, int32_t argc, const Janet *argv) {
|
||||||
|
peg_fixarity(b, argc, 2);
|
||||||
|
Reserve r = reserve(b, 3);
|
||||||
|
uint32_t subrule1 = peg_compile1(b, argv[0]);
|
||||||
|
uint32_t subrule2 = peg_compile1(b, argv[1]);
|
||||||
|
emit_2(r, RULE_TIL, subrule1, subrule2);
|
||||||
|
}
|
||||||
|
|
||||||
static void spec_split(Builder *b, int32_t argc, const Janet *argv) {
|
static void spec_split(Builder *b, int32_t argc, const Janet *argv) {
|
||||||
peg_fixarity(b, argc, 2);
|
peg_fixarity(b, argc, 2);
|
||||||
Reserve r = reserve(b, 3);
|
Reserve r = reserve(b, 3);
|
||||||
@@ -1323,6 +1370,7 @@ static const SpecialPair peg_specials[] = {
|
|||||||
{"split", spec_split},
|
{"split", spec_split},
|
||||||
{"sub", spec_sub},
|
{"sub", spec_sub},
|
||||||
{"thru", spec_thru},
|
{"thru", spec_thru},
|
||||||
|
{"til", spec_til},
|
||||||
{"to", spec_to},
|
{"to", spec_to},
|
||||||
{"uint", spec_uint_le},
|
{"uint", spec_uint_le},
|
||||||
{"uint-be", spec_uint_be},
|
{"uint-be", spec_uint_be},
|
||||||
@@ -1416,6 +1464,11 @@ static uint32_t peg_compile1(Builder *b, Janet peg) {
|
|||||||
emit_bytes(b, RULE_LITERAL, len, str);
|
emit_bytes(b, RULE_LITERAL, len, str);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case JANET_BUFFER: {
|
||||||
|
const JanetBuffer *buf = janet_unwrap_buffer(peg);
|
||||||
|
emit_bytes(b, RULE_LITERAL, buf->count, buf->data);
|
||||||
|
break;
|
||||||
|
}
|
||||||
case JANET_TABLE: {
|
case JANET_TABLE: {
|
||||||
/* Build grammar table */
|
/* Build grammar table */
|
||||||
JanetTable *new_grammar = janet_table_clone(janet_unwrap_table(peg));
|
JanetTable *new_grammar = janet_table_clone(janet_unwrap_table(peg));
|
||||||
@@ -1657,6 +1710,7 @@ static void *peg_unmarshal(JanetMarshalContext *ctx) {
|
|||||||
i += 4;
|
i += 4;
|
||||||
break;
|
break;
|
||||||
case RULE_SUB:
|
case RULE_SUB:
|
||||||
|
case RULE_TIL:
|
||||||
case RULE_SPLIT:
|
case RULE_SPLIT:
|
||||||
/* [rule, rule] */
|
/* [rule, rule] */
|
||||||
if (rule[1] >= blen) goto bad;
|
if (rule[1] >= blen) goto bad;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -1066,7 +1066,7 @@ void janet_buffer_format(
|
|||||||
if (form[2] == '\0')
|
if (form[2] == '\0')
|
||||||
janet_buffer_push_bytes(b, s, l);
|
janet_buffer_push_bytes(b, s, l);
|
||||||
else {
|
else {
|
||||||
if (l != (int32_t) strlen((const char *) s))
|
if (l != (int32_t) strnlen((const char *) s, l))
|
||||||
janet_panic("string contains zeros");
|
janet_panic("string contains zeros");
|
||||||
if (!strchr(form, '.') && l >= 100) {
|
if (!strchr(form, '.') && l >= 100) {
|
||||||
janet_panic("no precision and string is too long to be formatted");
|
janet_panic("no precision and string is too long to be formatted");
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -28,7 +28,7 @@
|
|||||||
|
|
||||||
/* Run a string */
|
/* Run a string */
|
||||||
int janet_dobytes(JanetTable *env, const uint8_t *bytes, int32_t len, const char *sourcePath, Janet *out) {
|
int janet_dobytes(JanetTable *env, const uint8_t *bytes, int32_t len, const char *sourcePath, Janet *out) {
|
||||||
JanetParser parser;
|
JanetParser *parser;
|
||||||
int errflags = 0, done = 0;
|
int errflags = 0, done = 0;
|
||||||
int32_t index = 0;
|
int32_t index = 0;
|
||||||
Janet ret = janet_wrap_nil();
|
Janet ret = janet_wrap_nil();
|
||||||
@@ -37,14 +37,16 @@ int janet_dobytes(JanetTable *env, const uint8_t *bytes, int32_t len, const char
|
|||||||
|
|
||||||
if (where) janet_gcroot(janet_wrap_string(where));
|
if (where) janet_gcroot(janet_wrap_string(where));
|
||||||
if (NULL == sourcePath) sourcePath = "<unknown>";
|
if (NULL == sourcePath) sourcePath = "<unknown>";
|
||||||
janet_parser_init(&parser);
|
parser = janet_abstract(&janet_parser_type, sizeof(JanetParser));
|
||||||
|
janet_parser_init(parser);
|
||||||
|
janet_gcroot(janet_wrap_abstract(parser));
|
||||||
|
|
||||||
/* While we haven't seen an error */
|
/* While we haven't seen an error */
|
||||||
while (!done) {
|
while (!done) {
|
||||||
|
|
||||||
/* Evaluate parsed values */
|
/* Evaluate parsed values */
|
||||||
while (janet_parser_has_more(&parser)) {
|
while (janet_parser_has_more(parser)) {
|
||||||
Janet form = janet_parser_produce(&parser);
|
Janet form = janet_parser_produce(parser);
|
||||||
JanetCompileResult cres = janet_compile(form, env, where);
|
JanetCompileResult cres = janet_compile(form, env, where);
|
||||||
if (cres.status == JANET_COMPILE_OK) {
|
if (cres.status == JANET_COMPILE_OK) {
|
||||||
JanetFunction *f = janet_thunk(cres.funcdef);
|
JanetFunction *f = janet_thunk(cres.funcdef);
|
||||||
@@ -58,8 +60,8 @@ int janet_dobytes(JanetTable *env, const uint8_t *bytes, int32_t len, const char
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
ret = janet_wrap_string(cres.error);
|
ret = janet_wrap_string(cres.error);
|
||||||
int32_t line = (int32_t) parser.line;
|
int32_t line = (int32_t) parser->line;
|
||||||
int32_t col = (int32_t) parser.column;
|
int32_t col = (int32_t) parser->column;
|
||||||
if ((cres.error_mapping.line > 0) &&
|
if ((cres.error_mapping.line > 0) &&
|
||||||
(cres.error_mapping.column > 0)) {
|
(cres.error_mapping.column > 0)) {
|
||||||
line = cres.error_mapping.line;
|
line = cres.error_mapping.line;
|
||||||
@@ -81,16 +83,16 @@ int janet_dobytes(JanetTable *env, const uint8_t *bytes, int32_t len, const char
|
|||||||
if (done) break;
|
if (done) break;
|
||||||
|
|
||||||
/* Dispatch based on parse state */
|
/* Dispatch based on parse state */
|
||||||
switch (janet_parser_status(&parser)) {
|
switch (janet_parser_status(parser)) {
|
||||||
case JANET_PARSE_DEAD:
|
case JANET_PARSE_DEAD:
|
||||||
done = 1;
|
done = 1;
|
||||||
break;
|
break;
|
||||||
case JANET_PARSE_ERROR: {
|
case JANET_PARSE_ERROR: {
|
||||||
const char *e = janet_parser_error(&parser);
|
const char *e = janet_parser_error(parser);
|
||||||
errflags |= 0x04;
|
errflags |= 0x04;
|
||||||
ret = janet_cstringv(e);
|
ret = janet_cstringv(e);
|
||||||
int32_t line = (int32_t) parser.line;
|
int32_t line = (int32_t) parser->line;
|
||||||
int32_t col = (int32_t) parser.column;
|
int32_t col = (int32_t) parser->column;
|
||||||
janet_eprintf("%s:%d:%d: parse error: %s\n", sourcePath, line, col, e);
|
janet_eprintf("%s:%d:%d: parse error: %s\n", sourcePath, line, col, e);
|
||||||
done = 1;
|
done = 1;
|
||||||
break;
|
break;
|
||||||
@@ -98,9 +100,9 @@ int janet_dobytes(JanetTable *env, const uint8_t *bytes, int32_t len, const char
|
|||||||
case JANET_PARSE_ROOT:
|
case JANET_PARSE_ROOT:
|
||||||
case JANET_PARSE_PENDING:
|
case JANET_PARSE_PENDING:
|
||||||
if (index >= len) {
|
if (index >= len) {
|
||||||
janet_parser_eof(&parser);
|
janet_parser_eof(parser);
|
||||||
} else {
|
} else {
|
||||||
janet_parser_consume(&parser, bytes[index++]);
|
janet_parser_consume(parser, bytes[index++]);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -108,7 +110,7 @@ int janet_dobytes(JanetTable *env, const uint8_t *bytes, int32_t len, const char
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Clean up and return errors */
|
/* Clean up and return errors */
|
||||||
janet_parser_deinit(&parser);
|
janet_gcunroot(janet_wrap_abstract(parser));
|
||||||
if (where) janet_gcunroot(janet_wrap_string(where));
|
if (where) janet_gcunroot(janet_wrap_string(where));
|
||||||
#ifdef JANET_EV
|
#ifdef JANET_EV
|
||||||
/* Enter the event loop if we are not already in it */
|
/* Enter the event loop if we are not already in it */
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -27,7 +27,9 @@
|
|||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
#ifdef JANET_EV
|
#ifdef JANET_EV
|
||||||
#ifndef JANET_WINDOWS
|
#ifdef JANET_WINDOWS
|
||||||
|
#include <windows.h>
|
||||||
|
#else
|
||||||
#include <pthread.h>
|
#include <pthread.h>
|
||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
@@ -53,13 +55,21 @@ typedef struct {
|
|||||||
void *data;
|
void *data;
|
||||||
} JanetQueue;
|
} JanetQueue;
|
||||||
|
|
||||||
|
#ifdef JANET_EV
|
||||||
typedef struct {
|
typedef struct {
|
||||||
JanetTimestamp when;
|
JanetTimestamp when;
|
||||||
JanetFiber *fiber;
|
JanetFiber *fiber;
|
||||||
JanetFiber *curr_fiber;
|
JanetFiber *curr_fiber;
|
||||||
uint32_t sched_id;
|
uint32_t sched_id;
|
||||||
int is_error;
|
int is_error;
|
||||||
|
int has_worker;
|
||||||
|
#ifdef JANET_WINDOWS
|
||||||
|
HANDLE worker;
|
||||||
|
#else
|
||||||
|
pthread_t worker;
|
||||||
|
#endif
|
||||||
} JanetTimeout;
|
} JanetTimeout;
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Registry table for C functions - contains metadata that can
|
/* Registry table for C functions - contains metadata that can
|
||||||
* be looked up by cfunction pointer. All strings here are pointing to
|
* be looked up by cfunction pointer. All strings here are pointing to
|
||||||
@@ -100,6 +110,7 @@ struct JanetVM {
|
|||||||
* return point for panics. */
|
* return point for panics. */
|
||||||
jmp_buf *signal_buf;
|
jmp_buf *signal_buf;
|
||||||
Janet *return_reg;
|
Janet *return_reg;
|
||||||
|
int coerce_error;
|
||||||
|
|
||||||
/* The global registry for c functions. Used to store meta-data
|
/* The global registry for c functions. Used to store meta-data
|
||||||
* along with otherwise bare c function pointers. */
|
* along with otherwise bare c function pointers. */
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -301,6 +301,7 @@ int janet_scan_number_base(
|
|||||||
if (base == 0) {
|
if (base == 0) {
|
||||||
base = 10;
|
base = 10;
|
||||||
}
|
}
|
||||||
|
int exp_base = base;
|
||||||
|
|
||||||
/* Skip leading zeros */
|
/* Skip leading zeros */
|
||||||
while (str < end && (*str == '0' || *str == '.')) {
|
while (str < end && (*str == '0' || *str == '.')) {
|
||||||
@@ -322,6 +323,12 @@ int janet_scan_number_base(
|
|||||||
} else if (*str == '&') {
|
} else if (*str == '&') {
|
||||||
foundexp = 1;
|
foundexp = 1;
|
||||||
break;
|
break;
|
||||||
|
} else if (base == 16 && (*str == 'P' || *str == 'p')) { /* IEEE hex float */
|
||||||
|
foundexp = 1;
|
||||||
|
exp_base = 10;
|
||||||
|
base = 2;
|
||||||
|
ex *= 4; /* We need to correct the current exponent after we change the base */
|
||||||
|
break;
|
||||||
} else if (base == 10 && (*str == 'E' || *str == 'e')) {
|
} else if (base == 10 && (*str == 'E' || *str == 'e')) {
|
||||||
foundexp = 1;
|
foundexp = 1;
|
||||||
break;
|
break;
|
||||||
@@ -360,9 +367,9 @@ int janet_scan_number_base(
|
|||||||
}
|
}
|
||||||
while (str < end) {
|
while (str < end) {
|
||||||
int digit = digit_lookup[*str & 0x7F];
|
int digit = digit_lookup[*str & 0x7F];
|
||||||
if (*str > 127 || digit >= base) goto error;
|
if (*str > 127 || digit >= exp_base) goto error;
|
||||||
if (ee < (INT32_MAX / 40)) {
|
if (ee < (INT32_MAX / 40)) {
|
||||||
ee = base * ee + digit;
|
ee = exp_base * ee + digit;
|
||||||
}
|
}
|
||||||
str++;
|
str++;
|
||||||
seenadigit = 1;
|
seenadigit = 1;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -294,6 +294,16 @@ JANET_CORE_FN(cfun_struct_to_table,
|
|||||||
return janet_wrap_table(tab);
|
return janet_wrap_table(tab);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
JANET_CORE_FN(cfun_struct_rawget,
|
||||||
|
"(struct/rawget st key)",
|
||||||
|
"Gets a value from a struct `st` without looking at the prototype struct. "
|
||||||
|
"If `st` does not contain the key directly, the function will return "
|
||||||
|
"nil without checking the prototype. Returns the value in the struct.") {
|
||||||
|
janet_fixarity(argc, 2);
|
||||||
|
JanetStruct st = janet_getstruct(argv, 0);
|
||||||
|
return janet_struct_rawget(st, argv[1]);
|
||||||
|
}
|
||||||
|
|
||||||
/* Load the struct module */
|
/* Load the struct module */
|
||||||
void janet_lib_struct(JanetTable *env) {
|
void janet_lib_struct(JanetTable *env) {
|
||||||
JanetRegExt struct_cfuns[] = {
|
JanetRegExt struct_cfuns[] = {
|
||||||
@@ -301,6 +311,7 @@ void janet_lib_struct(JanetTable *env) {
|
|||||||
JANET_CORE_REG("struct/getproto", cfun_struct_getproto),
|
JANET_CORE_REG("struct/getproto", cfun_struct_getproto),
|
||||||
JANET_CORE_REG("struct/proto-flatten", cfun_struct_flatten),
|
JANET_CORE_REG("struct/proto-flatten", cfun_struct_flatten),
|
||||||
JANET_CORE_REG("struct/to-table", cfun_struct_to_table),
|
JANET_CORE_REG("struct/to-table", cfun_struct_to_table),
|
||||||
|
JANET_CORE_REG("struct/rawget", cfun_struct_rawget),
|
||||||
JANET_REG_END
|
JANET_REG_END
|
||||||
};
|
};
|
||||||
janet_core_cfuns_ext(env, NULL, struct_cfuns);
|
janet_core_cfuns_ext(env, NULL, struct_cfuns);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -372,12 +372,14 @@ JANET_CORE_FN(cfun_table_setproto,
|
|||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(cfun_table_tostruct,
|
JANET_CORE_FN(cfun_table_tostruct,
|
||||||
"(table/to-struct tab)",
|
"(table/to-struct tab &opt proto)",
|
||||||
"Convert a table to a struct. Returns a new struct. This function "
|
"Convert a table to a struct. Returns a new struct.") {
|
||||||
"does not take into account prototype tables.") {
|
janet_arity(argc, 1, 2);
|
||||||
janet_fixarity(argc, 1);
|
|
||||||
JanetTable *t = janet_gettable(argv, 0);
|
JanetTable *t = janet_gettable(argv, 0);
|
||||||
return janet_wrap_struct(janet_table_to_struct(t));
|
JanetStruct proto = janet_optstruct(argv, argc, 1, NULL);
|
||||||
|
JanetStruct st = janet_table_to_struct(t);
|
||||||
|
janet_struct_proto(st) = proto;
|
||||||
|
return janet_wrap_struct(st);
|
||||||
}
|
}
|
||||||
|
|
||||||
JANET_CORE_FN(cfun_table_rawget,
|
JANET_CORE_FN(cfun_table_rawget,
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -931,27 +931,24 @@ int janet_gettime(struct timespec *spec, enum JanetTimeSource source) {
|
|||||||
#include <mach/clock.h>
|
#include <mach/clock.h>
|
||||||
#include <mach/mach.h>
|
#include <mach/mach.h>
|
||||||
int janet_gettime(struct timespec *spec, enum JanetTimeSource source) {
|
int janet_gettime(struct timespec *spec, enum JanetTimeSource source) {
|
||||||
if (source == JANET_TIME_REALTIME) {
|
if (source == JANET_TIME_CPUTIME) {
|
||||||
|
clock_t tmp = clock();
|
||||||
|
spec->tv_sec = tmp / CLOCKS_PER_SEC;
|
||||||
|
spec->tv_nsec = ((tmp - (spec->tv_sec * CLOCKS_PER_SEC)) * 1000000000) / CLOCKS_PER_SEC;
|
||||||
|
} else {
|
||||||
clock_serv_t cclock;
|
clock_serv_t cclock;
|
||||||
mach_timespec_t mts;
|
mach_timespec_t mts;
|
||||||
host_get_clock_service(mach_host_self(), CALENDAR_CLOCK, &cclock);
|
clock_id_t cid = CALENDAR_CLOCK;
|
||||||
|
if (source == JANET_TIME_REALTIME) {
|
||||||
|
cid = CALENDAR_CLOCK;
|
||||||
|
} else if (source == JANET_TIME_MONOTONIC) {
|
||||||
|
cid = SYSTEM_CLOCK;
|
||||||
|
}
|
||||||
|
host_get_clock_service(mach_host_self(), cid, &cclock);
|
||||||
clock_get_time(cclock, &mts);
|
clock_get_time(cclock, &mts);
|
||||||
mach_port_deallocate(mach_task_self(), cclock);
|
mach_port_deallocate(mach_task_self(), cclock);
|
||||||
spec->tv_sec = mts.tv_sec;
|
spec->tv_sec = mts.tv_sec;
|
||||||
spec->tv_nsec = mts.tv_nsec;
|
spec->tv_nsec = mts.tv_nsec;
|
||||||
} else if (source == JANET_TIME_MONOTONIC) {
|
|
||||||
clock_serv_t cclock;
|
|
||||||
int nsecs;
|
|
||||||
mach_msg_type_number_t count;
|
|
||||||
host_get_clock_service(mach_host_self(), clock, &cclock);
|
|
||||||
clock_get_attributes(cclock, CLOCK_GET_TIME_RES, (clock_attr_t)&nsecs, &count);
|
|
||||||
mach_port_deallocate(mach_task_self(), cclock);
|
|
||||||
clock_getres(CLOCK_MONOTONIC, spec);
|
|
||||||
}
|
|
||||||
if (source == JANET_TIME_CPUTIME) {
|
|
||||||
clock_t tmp = clock();
|
|
||||||
spec->tv_sec = tmp;
|
|
||||||
spec->tv_nsec = (tmp - spec->tv_sec) * 1.0e9;
|
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -205,9 +205,9 @@ int janet_make_pipe(JanetHandle handles[2], int mode);
|
|||||||
#ifdef JANET_FILEWATCH
|
#ifdef JANET_FILEWATCH
|
||||||
void janet_lib_filewatch(JanetTable *env);
|
void janet_lib_filewatch(JanetTable *env);
|
||||||
#endif
|
#endif
|
||||||
|
#endif
|
||||||
#ifdef JANET_FFI
|
#ifdef JANET_FFI
|
||||||
void janet_lib_ffi(JanetTable *env);
|
void janet_lib_ffi(JanetTable *env);
|
||||||
#endif
|
#endif
|
||||||
#endif
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -115,7 +115,7 @@
|
|||||||
#define vm_maybe_auto_suspend(COND)
|
#define vm_maybe_auto_suspend(COND)
|
||||||
#else
|
#else
|
||||||
#define vm_maybe_auto_suspend(COND) do { \
|
#define vm_maybe_auto_suspend(COND) do { \
|
||||||
if ((COND) && janet_vm.auto_suspend) { \
|
if ((COND) && janet_atomic_load_relaxed(&janet_vm.auto_suspend)) { \
|
||||||
fiber->flags |= (JANET_FIBER_RESUME_NO_USEVAL | JANET_FIBER_RESUME_NO_SKIP); \
|
fiber->flags |= (JANET_FIBER_RESUME_NO_USEVAL | JANET_FIBER_RESUME_NO_SKIP); \
|
||||||
vm_return(JANET_SIGNAL_INTERRUPT, janet_wrap_nil()); \
|
vm_return(JANET_SIGNAL_INTERRUPT, janet_wrap_nil()); \
|
||||||
} \
|
} \
|
||||||
@@ -798,14 +798,14 @@ static JanetSignal run_vm(JanetFiber *fiber, Janet in) {
|
|||||||
vm_pcnext();
|
vm_pcnext();
|
||||||
|
|
||||||
VM_OP(JOP_JUMP)
|
VM_OP(JOP_JUMP)
|
||||||
pc += DS;
|
|
||||||
vm_maybe_auto_suspend(DS <= 0);
|
vm_maybe_auto_suspend(DS <= 0);
|
||||||
|
pc += DS;
|
||||||
vm_next();
|
vm_next();
|
||||||
|
|
||||||
VM_OP(JOP_JUMP_IF)
|
VM_OP(JOP_JUMP_IF)
|
||||||
if (janet_truthy(stack[A])) {
|
if (janet_truthy(stack[A])) {
|
||||||
pc += ES;
|
|
||||||
vm_maybe_auto_suspend(ES <= 0);
|
vm_maybe_auto_suspend(ES <= 0);
|
||||||
|
pc += ES;
|
||||||
} else {
|
} else {
|
||||||
pc++;
|
pc++;
|
||||||
}
|
}
|
||||||
@@ -815,15 +815,15 @@ static JanetSignal run_vm(JanetFiber *fiber, Janet in) {
|
|||||||
if (janet_truthy(stack[A])) {
|
if (janet_truthy(stack[A])) {
|
||||||
pc++;
|
pc++;
|
||||||
} else {
|
} else {
|
||||||
pc += ES;
|
|
||||||
vm_maybe_auto_suspend(ES <= 0);
|
vm_maybe_auto_suspend(ES <= 0);
|
||||||
|
pc += ES;
|
||||||
}
|
}
|
||||||
vm_next();
|
vm_next();
|
||||||
|
|
||||||
VM_OP(JOP_JUMP_IF_NIL)
|
VM_OP(JOP_JUMP_IF_NIL)
|
||||||
if (janet_checktype(stack[A], JANET_NIL)) {
|
if (janet_checktype(stack[A], JANET_NIL)) {
|
||||||
pc += ES;
|
|
||||||
vm_maybe_auto_suspend(ES <= 0);
|
vm_maybe_auto_suspend(ES <= 0);
|
||||||
|
pc += ES;
|
||||||
} else {
|
} else {
|
||||||
pc++;
|
pc++;
|
||||||
}
|
}
|
||||||
@@ -833,8 +833,8 @@ static JanetSignal run_vm(JanetFiber *fiber, Janet in) {
|
|||||||
if (janet_checktype(stack[A], JANET_NIL)) {
|
if (janet_checktype(stack[A], JANET_NIL)) {
|
||||||
pc++;
|
pc++;
|
||||||
} else {
|
} else {
|
||||||
pc += ES;
|
|
||||||
vm_maybe_auto_suspend(ES <= 0);
|
vm_maybe_auto_suspend(ES <= 0);
|
||||||
|
pc += ES;
|
||||||
}
|
}
|
||||||
vm_next();
|
vm_next();
|
||||||
|
|
||||||
@@ -1373,7 +1373,10 @@ Janet janet_call(JanetFunction *fun, int32_t argc, const Janet *argv) {
|
|||||||
|
|
||||||
/* Run vm */
|
/* Run vm */
|
||||||
janet_vm.fiber->flags |= JANET_FIBER_RESUME_NO_USEVAL | JANET_FIBER_RESUME_NO_SKIP;
|
janet_vm.fiber->flags |= JANET_FIBER_RESUME_NO_USEVAL | JANET_FIBER_RESUME_NO_SKIP;
|
||||||
|
int old_coerce_error = janet_vm.coerce_error;
|
||||||
|
janet_vm.coerce_error = 1;
|
||||||
JanetSignal signal = run_vm(janet_vm.fiber, janet_wrap_nil());
|
JanetSignal signal = run_vm(janet_vm.fiber, janet_wrap_nil());
|
||||||
|
janet_vm.coerce_error = old_coerce_error;
|
||||||
|
|
||||||
/* Teardown */
|
/* Teardown */
|
||||||
janet_vm.stackn = oldn;
|
janet_vm.stackn = oldn;
|
||||||
@@ -1384,6 +1387,15 @@ Janet janet_call(JanetFunction *fun, int32_t argc, const Janet *argv) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (signal != JANET_SIGNAL_OK) {
|
if (signal != JANET_SIGNAL_OK) {
|
||||||
|
/* Should match logic in janet_signalv */
|
||||||
|
#ifdef JANET_EV
|
||||||
|
if (janet_vm.root_fiber != NULL && signal == JANET_SIGNAL_EVENT) {
|
||||||
|
janet_vm.root_fiber->sched_id++;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
if (signal != JANET_SIGNAL_ERROR) {
|
||||||
|
*janet_vm.return_reg = janet_wrap_string(janet_formatc("%v coerced from %s to error", *janet_vm.return_reg, janet_signal_names[signal]));
|
||||||
|
}
|
||||||
janet_panicv(*janet_vm.return_reg);
|
janet_panicv(*janet_vm.return_reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1430,8 +1442,10 @@ void janet_try_init(JanetTryState *state) {
|
|||||||
state->vm_fiber = janet_vm.fiber;
|
state->vm_fiber = janet_vm.fiber;
|
||||||
state->vm_jmp_buf = janet_vm.signal_buf;
|
state->vm_jmp_buf = janet_vm.signal_buf;
|
||||||
state->vm_return_reg = janet_vm.return_reg;
|
state->vm_return_reg = janet_vm.return_reg;
|
||||||
|
state->coerce_error = janet_vm.coerce_error;
|
||||||
janet_vm.return_reg = &(state->payload);
|
janet_vm.return_reg = &(state->payload);
|
||||||
janet_vm.signal_buf = &(state->buf);
|
janet_vm.signal_buf = &(state->buf);
|
||||||
|
janet_vm.coerce_error = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void janet_restore(JanetTryState *state) {
|
void janet_restore(JanetTryState *state) {
|
||||||
@@ -1440,6 +1454,7 @@ void janet_restore(JanetTryState *state) {
|
|||||||
janet_vm.fiber = state->vm_fiber;
|
janet_vm.fiber = state->vm_fiber;
|
||||||
janet_vm.signal_buf = state->vm_jmp_buf;
|
janet_vm.signal_buf = state->vm_jmp_buf;
|
||||||
janet_vm.return_reg = state->vm_return_reg;
|
janet_vm.return_reg = state->vm_return_reg;
|
||||||
|
janet_vm.coerce_error = state->coerce_error;
|
||||||
}
|
}
|
||||||
|
|
||||||
static JanetSignal janet_continue_no_check(JanetFiber *fiber, Janet in, Janet *out) {
|
static JanetSignal janet_continue_no_check(JanetFiber *fiber, Janet in, Janet *out) {
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
@@ -67,11 +67,21 @@ extern "C" {
|
|||||||
#define JANET_LINUX 1
|
#define JANET_LINUX 1
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Check for Android */
|
||||||
|
#ifdef __ANDROID__
|
||||||
|
#define JANET_ANDROID 1
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Check for Cygwin */
|
/* Check for Cygwin */
|
||||||
#if defined(__CYGWIN__)
|
#if defined(__CYGWIN__)
|
||||||
#define JANET_CYGWIN 1
|
#define JANET_CYGWIN 1
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Check for Illumos */
|
||||||
|
#if defined(__illumos__)
|
||||||
|
#define JANET_ILLUMOS 1
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Check Unix */
|
/* Check Unix */
|
||||||
#if defined(_AIX) \
|
#if defined(_AIX) \
|
||||||
|| defined(__APPLE__) /* Darwin */ \
|
|| defined(__APPLE__) /* Darwin */ \
|
||||||
@@ -157,7 +167,7 @@ extern "C" {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Check sun */
|
/* Check sun */
|
||||||
#ifdef __sun
|
#if defined(__sun) && !defined(JANET_ILLUMOS)
|
||||||
#define JANET_NO_UTC_MKTIME
|
#define JANET_NO_UTC_MKTIME
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -165,14 +175,12 @@ extern "C" {
|
|||||||
/* Also enable the thread library only if not single-threaded */
|
/* Also enable the thread library only if not single-threaded */
|
||||||
#ifdef JANET_SINGLE_THREADED
|
#ifdef JANET_SINGLE_THREADED
|
||||||
#define JANET_THREAD_LOCAL
|
#define JANET_THREAD_LOCAL
|
||||||
#undef JANET_THREADS
|
#elif !(defined(JANET_THREAD_LOCAL)) && defined(__GNUC__)
|
||||||
#elif defined(__GNUC__)
|
|
||||||
#define JANET_THREAD_LOCAL __thread
|
#define JANET_THREAD_LOCAL __thread
|
||||||
#elif defined(_MSC_BUILD)
|
#elif !(defined(JANET_THREAD_LOCAL)) && defined(_MSC_BUILD)
|
||||||
#define JANET_THREAD_LOCAL __declspec(thread)
|
#define JANET_THREAD_LOCAL __declspec(thread)
|
||||||
#else
|
#elif !(defined(JANET_THREAD_LOCAL))
|
||||||
#define JANET_THREAD_LOCAL
|
#define JANET_THREAD_LOCAL
|
||||||
#undef JANET_THREADS
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Enable or disable dynamic module loading. Enabled by default. */
|
/* Enable or disable dynamic module loading. Enabled by default. */
|
||||||
@@ -591,6 +599,7 @@ typedef void *JanetAbstract;
|
|||||||
#define JANET_STREAM_WRITABLE 0x400
|
#define JANET_STREAM_WRITABLE 0x400
|
||||||
#define JANET_STREAM_ACCEPTABLE 0x800
|
#define JANET_STREAM_ACCEPTABLE 0x800
|
||||||
#define JANET_STREAM_UDPSERVER 0x1000
|
#define JANET_STREAM_UDPSERVER 0x1000
|
||||||
|
#define JANET_STREAM_NOT_CLOSEABLE 0x2000
|
||||||
#define JANET_STREAM_TOCLOSE 0x10000
|
#define JANET_STREAM_TOCLOSE 0x10000
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
@@ -663,6 +672,7 @@ typedef int32_t JanetAtomicInt;
|
|||||||
JANET_API JanetAtomicInt janet_atomic_inc(JanetAtomicInt volatile *x);
|
JANET_API JanetAtomicInt janet_atomic_inc(JanetAtomicInt volatile *x);
|
||||||
JANET_API JanetAtomicInt janet_atomic_dec(JanetAtomicInt volatile *x);
|
JANET_API JanetAtomicInt janet_atomic_dec(JanetAtomicInt volatile *x);
|
||||||
JANET_API JanetAtomicInt janet_atomic_load(JanetAtomicInt volatile *x);
|
JANET_API JanetAtomicInt janet_atomic_load(JanetAtomicInt volatile *x);
|
||||||
|
JANET_API JanetAtomicInt janet_atomic_load_relaxed(JanetAtomicInt volatile *x);
|
||||||
|
|
||||||
/* We provide three possible implementations of Janets. The preferred
|
/* We provide three possible implementations of Janets. The preferred
|
||||||
* nanboxing approach, for 32 or 64 bits, and the standard C version. Code in the rest of the
|
* nanboxing approach, for 32 or 64 bits, and the standard C version. Code in the rest of the
|
||||||
@@ -1261,6 +1271,7 @@ typedef struct {
|
|||||||
/* new state */
|
/* new state */
|
||||||
jmp_buf buf;
|
jmp_buf buf;
|
||||||
Janet payload;
|
Janet payload;
|
||||||
|
int coerce_error;
|
||||||
} JanetTryState;
|
} JanetTryState;
|
||||||
|
|
||||||
/***** END SECTION TYPES *****/
|
/***** END SECTION TYPES *****/
|
||||||
@@ -2181,6 +2192,7 @@ typedef enum {
|
|||||||
RULE_UNREF, /* [rule, tag] */
|
RULE_UNREF, /* [rule, tag] */
|
||||||
RULE_CAPTURE_NUM, /* [rule, tag] */
|
RULE_CAPTURE_NUM, /* [rule, tag] */
|
||||||
RULE_SUB, /* [rule, rule] */
|
RULE_SUB, /* [rule, rule] */
|
||||||
|
RULE_TIL, /* [rule, rule] */
|
||||||
RULE_SPLIT, /* [rule, rule] */
|
RULE_SPLIT, /* [rule, rule] */
|
||||||
RULE_NTH, /* [nth, rule, tag] */
|
RULE_NTH, /* [nth, rule, tag] */
|
||||||
RULE_ONLY_TAGS, /* [rule] */
|
RULE_ONLY_TAGS, /* [rule] */
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024 Calvin Rose
|
* Copyright (c) 2025 Calvin Rose
|
||||||
*
|
*
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
* of this software and associated documentation files (the "Software"), to
|
* of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -39,7 +39,7 @@
|
|||||||
(defmacro assert
|
(defmacro assert
|
||||||
[x &opt e]
|
[x &opt e]
|
||||||
(def xx (gensym))
|
(def xx (gensym))
|
||||||
(default e ~',x)
|
(default e (string/format "%j" x))
|
||||||
~(do
|
~(do
|
||||||
(def ,xx ,x)
|
(def ,xx ,x)
|
||||||
(,assert-no-tail ,xx ,e)
|
(,assert-no-tail ,xx ,e)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
@@ -865,6 +865,13 @@
|
|||||||
(assert (deep= ~(,import* "a" :as "b" :fresh maybe)
|
(assert (deep= ~(,import* "a" :as "b" :fresh maybe)
|
||||||
(macex '(import a :as b :fresh maybe))) "import macro 2")
|
(macex '(import a :as b :fresh maybe))) "import macro 2")
|
||||||
|
|
||||||
|
# 2af3f21d
|
||||||
|
(assert-error "import macro 2" (macex '(import a :fresh)))
|
||||||
|
(assert-error "import macro 3" (macex '(import a :as b :fresh)))
|
||||||
|
(assert-error "import macro 4" (macex '(import b "notakeyword" value)))
|
||||||
|
(assert (deep= ~(,import* "a" :fresh nil)
|
||||||
|
(macex '(import a :fresh nil))) "import macro 5")
|
||||||
|
|
||||||
# #477 walk preserving bracket type
|
# #477 walk preserving bracket type
|
||||||
# 0a1d902f4
|
# 0a1d902f4
|
||||||
(assert (= :brackets (tuple/type (postwalk identity '[])))
|
(assert (= :brackets (tuple/type (postwalk identity '[])))
|
||||||
@@ -896,11 +903,18 @@
|
|||||||
(struct/with-proto {:a [1 2 3]} :c 22 :b [1 2 3 4] :d "test" :e "test2"))
|
(struct/with-proto {:a [1 2 3]} :c 22 :b [1 2 3 4] :d "test" :e "test2"))
|
||||||
(table/setproto table-to-freeze @{:a @[1 2 3]})
|
(table/setproto table-to-freeze @{:a @[1 2 3]})
|
||||||
|
|
||||||
(assert (deep= {:a [1 2 3] :b [1 2 3 4] :c 22 :d "test" :e "test2"}
|
(assert (deep= struct-to-thaw (freeze table-to-freeze)))
|
||||||
(freeze table-to-freeze)))
|
|
||||||
(assert (deep= table-to-freeze-with-inline-proto (thaw table-to-freeze)))
|
(assert (deep= table-to-freeze-with-inline-proto (thaw table-to-freeze)))
|
||||||
(assert (deep= table-to-freeze-with-inline-proto (thaw struct-to-thaw)))
|
(assert (deep= table-to-freeze-with-inline-proto (thaw struct-to-thaw)))
|
||||||
|
|
||||||
|
# Check that freezing mutable keys is deterministic
|
||||||
|
# for issue #1535
|
||||||
|
(def hashes @{})
|
||||||
|
(repeat 200
|
||||||
|
(def x (freeze {@"" 1 @"" 2 @"" 3 @"" 4 @"" 5}))
|
||||||
|
(put hashes (hash x) true))
|
||||||
|
(assert (= 1 (length hashes)) "freeze mutable keys is deterministic")
|
||||||
|
|
||||||
# Make sure Carriage Returns don't end up in doc strings
|
# Make sure Carriage Returns don't end up in doc strings
|
||||||
# e528b86
|
# e528b86
|
||||||
(assert (not (string/find "\r"
|
(assert (not (string/find "\r"
|
||||||
@@ -995,4 +1009,18 @@
|
|||||||
(assert-error "assertf error 3" (assertf false "%s message" "mystery"))
|
(assert-error "assertf error 3" (assertf false "%s message" "mystery"))
|
||||||
(assert-error "assertf error 4" (assertf nil "%s %s" "alice" "bob"))
|
(assert-error "assertf error 4" (assertf nil "%s %s" "alice" "bob"))
|
||||||
|
|
||||||
|
# issue #1535
|
||||||
|
(loop [i :range [1 1000]]
|
||||||
|
(assert (deep-not= @{:key1 "value1" @"key" "value2"}
|
||||||
|
@{:key1 "value1" @"key" "value2"}) "deep= mutable keys"))
|
||||||
|
(assert (deep-not= {"abc" 123} {@"abc" 123}) "deep= mutable keys vs immutable key")
|
||||||
|
(assert (deep-not= {@"" 1 @"" 2 @"" 3} {@"" 1 @"" 2 @"" 3}) "deep= duplicate mutable keys")
|
||||||
|
(assert (deep-not= {@"" @"" @"" @"" @"" 3} {@"" @"" @"" @"" @"" 3}) "deep= duplicate mutable keys 2")
|
||||||
|
(assert (deep-not= {@[] @"" @[] @"" @[] 3} {@[] @"" @[] @"" @[] 3}) "deep= duplicate mutable keys 3")
|
||||||
|
(assert (deep-not= {@{} @"" @{} @"" @{} 3} {@{} @"" @{} @"" @{} 3}) "deep= duplicate mutable keys 4")
|
||||||
|
(assert (deep-not= @{:key1 "value1" @"key2" @"value2"}
|
||||||
|
@{:key1 "value1" @"key2" "value2"}) "deep= mutable keys")
|
||||||
|
(assert (deep-not= @{:key1 "value1" [@"key2"] @"value2"}
|
||||||
|
@{:key1 "value1" [@"key2"] @"value2"}) "deep= mutable keys")
|
||||||
|
|
||||||
(end-suite)
|
(end-suite)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2024 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2024 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
@@ -174,6 +174,7 @@
|
|||||||
(assert (deep= (range 0 17 4) @[0 4 8 12 16]) "(range 0 17 4)")
|
(assert (deep= (range 0 17 4) @[0 4 8 12 16]) "(range 0 17 4)")
|
||||||
(assert (deep= (range 16 0 -4) @[16 12 8 4]) "(range 16 0 -4)")
|
(assert (deep= (range 16 0 -4) @[16 12 8 4]) "(range 16 0 -4)")
|
||||||
(assert (deep= (range 17 0 -4) @[17 13 9 5 1]) "(range 17 0 -4)")
|
(assert (deep= (range 17 0 -4) @[17 13 9 5 1]) "(range 17 0 -4)")
|
||||||
|
(assert-error "large range" (range 0xFFFFFFFFFF))
|
||||||
|
|
||||||
(assert (= (length (range 10)) 10) "(range 10)")
|
(assert (= (length (range 10)) 10) "(range 10)")
|
||||||
(assert (= (length (range -10)) 0) "(range -10)")
|
(assert (= (length (range -10)) 0) "(range -10)")
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose & contributors
|
# Copyright (c) 2025 Calvin Rose & contributors
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
@@ -106,6 +106,8 @@
|
|||||||
(calc-2 "(+ 9 10 11 12)"))
|
(calc-2 "(+ 9 10 11 12)"))
|
||||||
@[10 26 42]) "parallel subprocesses 2")
|
@[10 26 42]) "parallel subprocesses 2")
|
||||||
|
|
||||||
|
# (print "file piping")
|
||||||
|
|
||||||
# File piping
|
# File piping
|
||||||
# a1cc5ca04
|
# a1cc5ca04
|
||||||
(assert-no-error "file writing 1"
|
(assert-no-error "file writing 1"
|
||||||
@@ -199,7 +201,7 @@
|
|||||||
(assert s "made server 1")
|
(assert s "made server 1")
|
||||||
|
|
||||||
(defn test-echo [msg]
|
(defn test-echo [msg]
|
||||||
(with [conn (net/connect test-host test-port)]
|
(with [conn (assert (net/connect test-host test-port))]
|
||||||
(net/write conn msg)
|
(net/write conn msg)
|
||||||
(def res (net/read conn 1024))
|
(def res (net/read conn 1024))
|
||||||
(assert (= (string res) msg) (string "echo " msg))))
|
(assert (= (string res) msg) (string "echo " msg))))
|
||||||
@@ -213,6 +215,7 @@
|
|||||||
|
|
||||||
# Test on both server and client
|
# Test on both server and client
|
||||||
# 504411e
|
# 504411e
|
||||||
|
(var iterations 0)
|
||||||
(defn names-handler
|
(defn names-handler
|
||||||
[stream]
|
[stream]
|
||||||
(defer (:close stream)
|
(defer (:close stream)
|
||||||
@@ -220,21 +223,28 @@
|
|||||||
(ev/read stream 1)
|
(ev/read stream 1)
|
||||||
(def [host port] (net/localname stream))
|
(def [host port] (net/localname stream))
|
||||||
(assert (= host test-host) "localname host server")
|
(assert (= host test-host) "localname host server")
|
||||||
(assert (= port (scan-number test-port)) "localname port server")))
|
(assert (= port (scan-number test-port)) "localname port server")
|
||||||
|
(++ iterations)
|
||||||
|
(ev/write stream " ")))
|
||||||
|
|
||||||
|
# (print "local name / peer name testing")
|
||||||
|
|
||||||
# Test localname and peername
|
# Test localname and peername
|
||||||
# 077bf5eba
|
# 077bf5eba
|
||||||
(repeat 10
|
(repeat 10
|
||||||
(with [s (net/server test-host test-port names-handler)]
|
(with [s (net/server test-host test-port names-handler)]
|
||||||
(repeat 10
|
(repeat 10
|
||||||
(with [conn (net/connect test-host test-port)]
|
(with [conn (assert (net/connect test-host test-port))]
|
||||||
(def [host port] (net/peername conn))
|
(def [host port] (net/peername conn))
|
||||||
(assert (= host test-host) "peername host client ")
|
(assert (= host test-host) "peername host client ")
|
||||||
(assert (= port (scan-number test-port)) "peername port client")
|
(assert (= port (scan-number test-port)) "peername port client")
|
||||||
# let server close
|
(++ iterations)
|
||||||
(ev/write conn " "))))
|
(ev/write conn " ")
|
||||||
|
(ev/read conn 1))))
|
||||||
(gccollect))
|
(gccollect))
|
||||||
|
|
||||||
|
(assert (= iterations 200) "localname and peername not enough checks")
|
||||||
|
|
||||||
# Create pipe
|
# Create pipe
|
||||||
# 12f09ad2d
|
# 12f09ad2d
|
||||||
(var pipe-counter 0)
|
(var pipe-counter 0)
|
||||||
@@ -401,6 +411,8 @@
|
|||||||
(while (def msg (ev/read connection 100))
|
(while (def msg (ev/read connection 100))
|
||||||
(broadcast name (string msg)))))))
|
(broadcast name (string msg)))))))
|
||||||
|
|
||||||
|
# (print "chat app testing")
|
||||||
|
|
||||||
# Now launch the chat server
|
# Now launch the chat server
|
||||||
(def chat-server (net/listen test-host test-port))
|
(def chat-server (net/listen test-host test-port))
|
||||||
(ev/spawn
|
(ev/spawn
|
||||||
@@ -410,6 +422,10 @@
|
|||||||
(ev/call handler connection)
|
(ev/call handler connection)
|
||||||
(break))))
|
(break))))
|
||||||
|
|
||||||
|
# Make sure we can't bind again with no-reuse
|
||||||
|
(assert-error "no-reuse"
|
||||||
|
(net/listen test-host test-port :stream true))
|
||||||
|
|
||||||
# Read from socket
|
# Read from socket
|
||||||
|
|
||||||
(defn expect-read
|
(defn expect-read
|
||||||
@@ -418,11 +434,11 @@
|
|||||||
(assert (= result text) (string/format "expected %v, got %v" text result)))
|
(assert (= result text) (string/format "expected %v, got %v" text result)))
|
||||||
|
|
||||||
# Now do our telnet chat
|
# Now do our telnet chat
|
||||||
(def bob (net/connect test-host test-port))
|
(def bob (assert (net/connect test-host test-port :stream)))
|
||||||
(expect-read bob "Whats your name?\n")
|
(expect-read bob "Whats your name?\n")
|
||||||
(net/write bob "bob")
|
(net/write bob "bob")
|
||||||
(expect-read bob "Welcome bob\n")
|
(expect-read bob "Welcome bob\n")
|
||||||
(def alice (net/connect test-host test-port))
|
(def alice (assert (net/connect test-host test-port)))
|
||||||
(expect-read alice "Whats your name?\n")
|
(expect-read alice "Whats your name?\n")
|
||||||
(net/write alice "alice")
|
(net/write alice "alice")
|
||||||
(expect-read alice "Welcome alice\n")
|
(expect-read alice "Welcome alice\n")
|
||||||
@@ -436,7 +452,7 @@
|
|||||||
(expect-read bob "[alice]:hi\n")
|
(expect-read bob "[alice]:hi\n")
|
||||||
|
|
||||||
# Ted joins the chat server
|
# Ted joins the chat server
|
||||||
(def ted (net/connect test-host test-port))
|
(def ted (assert (net/connect test-host test-port)))
|
||||||
(expect-read ted "Whats your name?\n")
|
(expect-read ted "Whats your name?\n")
|
||||||
(net/write ted "ted")
|
(net/write ted "ted")
|
||||||
(expect-read ted "Welcome ted\n")
|
(expect-read ted "Welcome ted\n")
|
||||||
@@ -465,4 +481,126 @@
|
|||||||
# Close chat server
|
# Close chat server
|
||||||
(:close chat-server)
|
(:close chat-server)
|
||||||
|
|
||||||
|
# Issue #1531
|
||||||
|
(defn sleep-print [x] (ev/sleep 0) (print x))
|
||||||
|
(protect (with-dyns [*out* sleep-print] (prin :foo)))
|
||||||
|
(defn level-trigger-handling [conn &] (:close conn))
|
||||||
|
(def s (assert (net/server test-host test-port level-trigger-handling)))
|
||||||
|
(def c (assert (net/connect test-host test-port)))
|
||||||
|
(:close s)
|
||||||
|
|
||||||
|
# Issue #1531 no. 2
|
||||||
|
(def c (ev/chan 0))
|
||||||
|
(ev/spawn (while (def x (ev/take c))))
|
||||||
|
(defn print-to-chan [x] (ev/give c x))
|
||||||
|
(assert-error "coerce await inside janet_call to error"
|
||||||
|
(with-dyns [*out* print-to-chan]
|
||||||
|
(pp :foo)))
|
||||||
|
(ev/chan-close c)
|
||||||
|
|
||||||
|
# soreuseport on unix domain sockets
|
||||||
|
(compwhen (or (= :macos (os/which)) (= :linux (os/which)))
|
||||||
|
(assert-no-error "unix-domain socket reuseaddr"
|
||||||
|
(let [uds-path "./unix-domain-socket"]
|
||||||
|
(defer (os/rm uds-path)
|
||||||
|
(let [s (net/listen :unix uds-path :stream)]
|
||||||
|
(:close s))))))
|
||||||
|
|
||||||
|
# (print "accept loop testing")
|
||||||
|
|
||||||
|
# net/accept-loop level triggering
|
||||||
|
(gccollect)
|
||||||
|
(def maxconn 50)
|
||||||
|
(var connect-count 0)
|
||||||
|
(defn level-trigger-handling
|
||||||
|
[conn &]
|
||||||
|
(with [conn conn]
|
||||||
|
(ev/write conn (ev/read conn 4096))
|
||||||
|
(++ connect-count)))
|
||||||
|
(def s (assert (net/server test-host test-port level-trigger-handling)))
|
||||||
|
(def cons @[])
|
||||||
|
(repeat maxconn (array/push cons (assert (net/connect test-host test-port))))
|
||||||
|
(assert (= maxconn (length cons)))
|
||||||
|
(defn do-connect [i]
|
||||||
|
(with [c (get cons i)]
|
||||||
|
(ev/write c "abc123")
|
||||||
|
(ev/read c 4096)))
|
||||||
|
(for i 0 maxconn (ev/spawn (do-connect i)))
|
||||||
|
(ev/sleep 0.1)
|
||||||
|
(assert (= maxconn connect-count))
|
||||||
|
(:close s)
|
||||||
|
|
||||||
|
# (print "running deadline tests...")
|
||||||
|
|
||||||
|
# Cancel os/proc-wait with ev/deadline
|
||||||
|
(let [p (os/spawn [;run janet "-e" "(os/sleep 4)"] :p)]
|
||||||
|
(var terminated-normally false)
|
||||||
|
(assert-error "deadline expired"
|
||||||
|
(ev/with-deadline 0.01
|
||||||
|
(os/proc-wait p)
|
||||||
|
(print "uhoh")
|
||||||
|
(set terminated-normally true)))
|
||||||
|
(assert (not terminated-normally) "early termination failure")
|
||||||
|
# Without this kill, janet will wait the full 4 seconds for the subprocess to complete before exiting.
|
||||||
|
(assert-no-error "kill proc after wait failed" (os/proc-kill p)))
|
||||||
|
|
||||||
|
# Cancel os/proc-wait with ev/deadline 2
|
||||||
|
(let [p (os/spawn [;run janet "-e" "(os/sleep 0.1)"] :p)]
|
||||||
|
(var terminated-normally false)
|
||||||
|
(assert-error "deadline expired"
|
||||||
|
(ev/with-deadline 0.05
|
||||||
|
(os/proc-wait p)
|
||||||
|
(print "uhoh")
|
||||||
|
(set terminated-normally true)))
|
||||||
|
(assert (not terminated-normally) "early termination failure 2")
|
||||||
|
(ev/sleep 0.15)
|
||||||
|
(assert (not terminated-normally) "early termination failure 3"))
|
||||||
|
|
||||||
|
# Deadline with interrupt
|
||||||
|
(defmacro with-deadline2
|
||||||
|
``
|
||||||
|
Create a fiber to execute `body`, schedule the event loop to cancel
|
||||||
|
the task (root fiber) associated with `body`'s fiber, and start
|
||||||
|
`body`'s fiber by resuming it.
|
||||||
|
|
||||||
|
The event loop will try to cancel the root fiber if `body`'s fiber
|
||||||
|
has not completed after at least `sec` seconds.
|
||||||
|
|
||||||
|
`sec` is a number that can have a fractional part.
|
||||||
|
``
|
||||||
|
[sec & body]
|
||||||
|
(with-syms [f]
|
||||||
|
~(let [,f (coro ,;body)]
|
||||||
|
(,ev/deadline ,sec nil ,f true)
|
||||||
|
(,resume ,f))))
|
||||||
|
|
||||||
|
(for i 0 10
|
||||||
|
# (print "deadline 1 iteration " i)
|
||||||
|
(assert (= :done (with-deadline2 10
|
||||||
|
(ev/sleep 0.01)
|
||||||
|
:done)) "deadline with interrupt exits normally"))
|
||||||
|
|
||||||
|
(for i 0 10
|
||||||
|
# (print "deadline 2 iteration " i)
|
||||||
|
(let [f (coro (forever :foo))]
|
||||||
|
(ev/deadline 0.01 nil f true)
|
||||||
|
(assert-error "deadline expired" (resume f))))
|
||||||
|
|
||||||
|
# Use :err :stdout
|
||||||
|
(def- subproc-code '(do (eprint "hi") (eflush) (print "there") (flush)))
|
||||||
|
(defn ev/slurp
|
||||||
|
[f &opt buf]
|
||||||
|
(default buf @"")
|
||||||
|
(if (ev/read f 0x10000 buf)
|
||||||
|
(ev/slurp f buf)
|
||||||
|
buf))
|
||||||
|
(def p (os/spawn [;run janet "-e" (string/format "%j" subproc-code)] :px {:out :pipe :err :out}))
|
||||||
|
(def [exit-code data]
|
||||||
|
(ev/gather
|
||||||
|
(os/proc-wait p)
|
||||||
|
(ev/slurp (p :out))))
|
||||||
|
(def data (string/replace-all "\r" "" data))
|
||||||
|
(assert (zero? exit-code) "subprocess ran")
|
||||||
|
(assert (= data "hi\nthere\n") "output is correct")
|
||||||
|
|
||||||
(end-suite)
|
(end-suite)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose & contributors
|
# Copyright (c) 2025 Calvin Rose & contributors
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
@@ -55,4 +55,11 @@
|
|||||||
(compwhen has-ffi
|
(compwhen has-ffi
|
||||||
(assert-error "bad struct issue #1512" (ffi/struct :void)))
|
(assert-error "bad struct issue #1512" (ffi/struct :void)))
|
||||||
|
|
||||||
|
(compwhen has-ffi
|
||||||
|
(def buf @"")
|
||||||
|
(ffi/write :u8 10 buf)
|
||||||
|
(assert (= 1 (length buf)))
|
||||||
|
(ffi/write :u8 10 buf)
|
||||||
|
(assert (= 2 (length buf))))
|
||||||
|
|
||||||
(end-suite)
|
(end-suite)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2024 Calvin Rose & contributors
|
# Copyright (c) 2025 Calvin Rose & contributors
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose & contributors
|
# Copyright (c) 2025 Calvin Rose & contributors
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose & contributors
|
# Copyright (c) 2025 Calvin Rose & contributors
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
@@ -207,7 +207,7 @@ neldb\0\0\0\xD8\x05printG\x01\0\xDE\xDE\xDE'\x03\0marshal_tes/\x02
|
|||||||
(assert (= 2 (length tclone)) "table/weak-values marsh 2")
|
(assert (= 2 (length tclone)) "table/weak-values marsh 2")
|
||||||
(gccollect)
|
(gccollect)
|
||||||
(assert (= 1 (length t)) "table/weak-value marsh 3")
|
(assert (= 1 (length t)) "table/weak-value marsh 3")
|
||||||
(assert (deep= t tclone) "table/weak-values marsh 4")
|
(assert (deep= (freeze t) (freeze tclone)) "table/weak-values marsh 4")
|
||||||
|
|
||||||
# tables with prototypes
|
# tables with prototypes
|
||||||
(def t (table/weak-values 1))
|
(def t (table/weak-values 1))
|
||||||
@@ -219,7 +219,7 @@ neldb\0\0\0\xD8\x05printG\x01\0\xDE\xDE\xDE'\x03\0marshal_tes/\x02
|
|||||||
(assert (= 2 (length tclone)) "marsh weak tables with prototypes 2")
|
(assert (= 2 (length tclone)) "marsh weak tables with prototypes 2")
|
||||||
(gccollect)
|
(gccollect)
|
||||||
(assert (= 1 (length t)) "marsh weak tables with prototypes 3")
|
(assert (= 1 (length t)) "marsh weak tables with prototypes 3")
|
||||||
(assert (deep= t tclone) "marsh weak tables with prototypes 4")
|
(assert (deep= (freeze t) (freeze tclone)) "marsh weak tables with prototypes 4")
|
||||||
(assert (deep= (getproto t) (getproto tclone)) "marsh weak tables with prototypes 5")
|
(assert (deep= (getproto t) (getproto tclone)) "marsh weak tables with prototypes 5")
|
||||||
|
|
||||||
(end-suite)
|
(end-suite)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
@@ -57,6 +57,8 @@
|
|||||||
(for i (+ index 1) (+ index indent 1)
|
(for i (+ index 1) (+ index indent 1)
|
||||||
(case (get text i)
|
(case (get text i)
|
||||||
nil (break)
|
nil (break)
|
||||||
|
(chr "\r") (if-not (= (chr "\n") (get text (inc i)))
|
||||||
|
(set rewrite false))
|
||||||
(chr "\n") (break)
|
(chr "\n") (break)
|
||||||
(chr " ") nil
|
(chr " ") nil
|
||||||
(set rewrite false))))
|
(set rewrite false))))
|
||||||
@@ -64,12 +66,17 @@
|
|||||||
# Only re-indent if no dedented characters.
|
# Only re-indent if no dedented characters.
|
||||||
(def str
|
(def str
|
||||||
(if rewrite
|
(if rewrite
|
||||||
(peg/replace-all ~(* "\n" (between 0 ,indent " ")) "\n" text)
|
(peg/replace-all ~(* '(* (? "\r") "\n") (between 0 ,indent " "))
|
||||||
|
(fn [mtch eol] eol) text)
|
||||||
text))
|
text))
|
||||||
|
|
||||||
(def first-nl (= (chr "\n") (first str)))
|
(def first-eol (cond
|
||||||
(def last-nl (= (chr "\n") (last str)))
|
(string/has-prefix? "\r\n" str) :crlf
|
||||||
(string/slice str (if first-nl 1 0) (if last-nl -2)))
|
(string/has-prefix? "\n" str) :lf))
|
||||||
|
(def last-eol (cond
|
||||||
|
(string/has-suffix? "\r\n" str) :crlf
|
||||||
|
(string/has-suffix? "\n" str) :lf))
|
||||||
|
(string/slice str (case first-eol :crlf 2 :lf 1 0) (case last-eol :crlf -3 :lf -2)))
|
||||||
|
|
||||||
(defn reindent-reference
|
(defn reindent-reference
|
||||||
"Same as reindent but use parser functionality. Useful for
|
"Same as reindent but use parser functionality. Useful for
|
||||||
@@ -89,8 +96,10 @@
|
|||||||
(let [a (reindent text indent)
|
(let [a (reindent text indent)
|
||||||
b (reindent-reference text indent)]
|
b (reindent-reference text indent)]
|
||||||
(assert (= a b)
|
(assert (= a b)
|
||||||
(string "indent " indent-counter " (indent=" indent ")"))))
|
(string/format "reindent: %q, parse: %q (indent-test #%d with indent of %d)" a b indent-counter indent)
|
||||||
|
)))
|
||||||
|
|
||||||
|
# Unix EOLs
|
||||||
(check-indent "" 0)
|
(check-indent "" 0)
|
||||||
(check-indent "\n" 0)
|
(check-indent "\n" 0)
|
||||||
(check-indent "\n" 1)
|
(check-indent "\n" 1)
|
||||||
@@ -106,6 +115,17 @@
|
|||||||
(check-indent "\n Hello, world!\n " 4)
|
(check-indent "\n Hello, world!\n " 4)
|
||||||
(check-indent "\n Hello, world!\n dedented text\n " 4)
|
(check-indent "\n Hello, world!\n dedented text\n " 4)
|
||||||
(check-indent "\n Hello, world!\n indented text\n " 4)
|
(check-indent "\n Hello, world!\n indented text\n " 4)
|
||||||
|
# Windows EOLs
|
||||||
|
(check-indent "\r\n" 0)
|
||||||
|
(check-indent "\r\n" 1)
|
||||||
|
(check-indent "\r\n\r\n" 0)
|
||||||
|
(check-indent "\r\n\r\n" 1)
|
||||||
|
(check-indent "\r\nHello, world!" 0)
|
||||||
|
(check-indent "\r\nHello, world!" 1)
|
||||||
|
(check-indent "\r\n Hello, world!\r\n " 4)
|
||||||
|
(check-indent "\r\n Hello, world!\r\n " 4)
|
||||||
|
(check-indent "\r\n Hello, world!\r\n dedented text\r\n " 4)
|
||||||
|
(check-indent "\r\n Hello, world!\r\n indented text\r\n " 4)
|
||||||
|
|
||||||
# Symbols with @ character
|
# Symbols with @ character
|
||||||
# d68eae9
|
# d68eae9
|
||||||
@@ -188,5 +208,14 @@
|
|||||||
(parser/consume p `")`)
|
(parser/consume p `")`)
|
||||||
(assert (= (parser/produce p) ["hello"]))
|
(assert (= (parser/produce p) ["hello"]))
|
||||||
|
|
||||||
|
# Hex floats
|
||||||
|
(assert (= math/pi +0x1.921fb54442d18p+0001))
|
||||||
|
(assert (= math/int-max +0x1.ffff_ffff_ffff_ffp+0052))
|
||||||
|
(assert (= math/int-min -0x1.ffff_ffff_ffff_ffp+0052))
|
||||||
|
(assert (= 1 0x1P0))
|
||||||
|
(assert (= 2 0x1P1))
|
||||||
|
(assert (= -2 -0x1p1))
|
||||||
|
(assert (= -0.5 -0x1p-1))
|
||||||
|
|
||||||
(end-suite)
|
(end-suite)
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
@@ -713,6 +713,41 @@
|
|||||||
"abcdef"
|
"abcdef"
|
||||||
@[])
|
@[])
|
||||||
|
|
||||||
|
(test "til: basic matching"
|
||||||
|
~(til "d" "abc")
|
||||||
|
"abcdef"
|
||||||
|
@[])
|
||||||
|
|
||||||
|
(test "til: second pattern can't see past the first occurrence of first pattern"
|
||||||
|
~(til "d" (* "abc" -1))
|
||||||
|
"abcdef"
|
||||||
|
@[])
|
||||||
|
|
||||||
|
(test "til: fails if first pattern fails"
|
||||||
|
~(til "x" "abc")
|
||||||
|
"abcdef"
|
||||||
|
nil)
|
||||||
|
|
||||||
|
(test "til: fails if second pattern fails"
|
||||||
|
~(til "abc" "x")
|
||||||
|
"abcdef"
|
||||||
|
nil)
|
||||||
|
|
||||||
|
(test "til: discards captures from initial pattern"
|
||||||
|
~(til '"d" '"abc")
|
||||||
|
"abcdef"
|
||||||
|
@["abc"])
|
||||||
|
|
||||||
|
(test "til: positions inside second match are still relative to the entire input"
|
||||||
|
~(* "one\ntw" (til 0 (* ($) (line) (column))))
|
||||||
|
"one\ntwo\nthree\n"
|
||||||
|
@[6 2 3])
|
||||||
|
|
||||||
|
(test "til: advances to the end of the first pattern's first occurrence"
|
||||||
|
~(* (til "d" "ab") "e")
|
||||||
|
"abcdef"
|
||||||
|
@[])
|
||||||
|
|
||||||
(test "split: basic functionality"
|
(test "split: basic functionality"
|
||||||
~(split "," '1)
|
~(split "," '1)
|
||||||
"a,b,c"
|
"a,b,c"
|
||||||
@@ -772,5 +807,33 @@
|
|||||||
"5:apple6:banana6:cherry"
|
"5:apple6:banana6:cherry"
|
||||||
@["apple" "banana" "cherry"])
|
@["apple" "banana" "cherry"])
|
||||||
|
|
||||||
|
# Issue #1539 - make sure split with "" doesn't infinite loop/oom
|
||||||
|
(test "issue 1539"
|
||||||
|
~(split "" (capture (to -1)))
|
||||||
|
"hello there friends"
|
||||||
|
nil)
|
||||||
|
|
||||||
|
(test "issue 1539 pt. 2"
|
||||||
|
~(split "," (capture 0))
|
||||||
|
"abc123,,,,"
|
||||||
|
@["" "" "" "" ""])
|
||||||
|
|
||||||
|
# Issue #1549 - allow buffers as peg literals
|
||||||
|
(test "issue 1549"
|
||||||
|
''@"abc123"
|
||||||
|
"abc123"
|
||||||
|
@["abc123"])
|
||||||
|
|
||||||
|
# Issue 1554 - 0-width match termination behavior
|
||||||
|
(test "issue 1554 case 1" '(any (> '1)) "abc" @[])
|
||||||
|
(test "issue 1554 case 2" '(any (? (> '1))) "abc" @[])
|
||||||
|
(test "issue 1554 case 3" '(any (> (? '1))) "abc" @[])
|
||||||
|
(test "issue 1554 case 4" '(* "a" (> '1)) "abc" @["b"])
|
||||||
|
(test "issue 1554 case 5" '(* "a" (? (> '1))) "abc" @["b"])
|
||||||
|
(test "issue 1554 case 6" '(* "a" (> (? '1))) "abc" @["b"])
|
||||||
|
(test "issue 1554 case 7" '(between 0 2 (> '1)) "abc" @["a" "a"])
|
||||||
|
(test "issue 1554 case 8" '(between 2 3 (? (> '1))) "abc" @["a" "a" "a"])
|
||||||
|
(test "issue 1554 case 9" '(between 0 0 (> (? '1))) "abc" @[])
|
||||||
|
|
||||||
(end-suite)
|
(end-suite)
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose & contributors
|
# Copyright (c) 2025 Calvin Rose & contributors
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
# Copyright (c) 2023 Calvin Rose
|
# Copyright (c) 2025 Calvin Rose
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
# of this software and associated documentation files (the "Software"), to
|
# of this software and associated documentation files (the "Software"), to
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user