Comment: | add janet API bootstrapping infra (prim, core), begin building executor (vm.c), design class model, continue working on loading, add script instances, tidy up src organization, build janet to use godot malloc/free |
---|---|
Downloads: | Tarball | ZIP archive | SQL archive |
Timelines: | family | ancestors | descendants | both | trunk |
Files: | files | file ages | folders |
SHA3-256: |
91e02e35d553b5de0f64342b5a954a53 |
User & Date: | lexi on 2025-02-21 22:02:29 |
Other Links: | manifest | tags |
2025-02-28
| ||
00:10 | continue iterating on object model check-in: c0fd81ac3d user: lexi tags: trunk | |
2025-02-21
| ||
22:02 | add janet API bootstrapping infra (prim, core), begin building executor (vm.c), design class model, continue working on loading, add script instances, tidy up src organization, build janet to use godot malloc/free check-in: 91e02e35d5 user: lexi tags: trunk | |
2025-02-10
| ||
01:11 | add automatic instance bindings with accessors check-in: 351bb17bed user: lexi tags: trunk | |
Modified gdjn.ct from [af1bbb3860] to [a79a150ec2].
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
..
50
51
52
53
54
55
56
57
58
59
60
61
62
63
|
%toc
@civ {
to meet the bare minimum standards necessary to qualify as [!civilized] to my mind, a language must:
* support compilation, whether to bytecode or machine code (excludes gdscript)
* parse into an AST (excludes bash, php (or used to))
* support AST-rewriting macros (excludes everything outside the lisp family except maybe rust)
* use real syntax, not whitespace-based kludges (excludes python, gdscript inter alia)
* provide a proper module system (excludes C)
* provide for hygienic namespacing (excludes C, gdscript)
* provide simple, cheap aggregate types (excludes gdscript, java, maybe python depending on how you view tuples)
so, janet and fennel are about the only game in town.
}
................................................................................
abi: okay yes in reality it absolutely fucking does, everyone uses the same gods-damned amd one, in actual materially existing reality there is no interop problem at least on x86, but because we haven't ritually enshrined this model at ISO and pacified its marauding [!aramitama] with burnt offerings of motor oil and obsolete DRAM chips, we cover our eyes and pretend we cannot see what is sitting right in front of us. i love too compute dont u
so GDExtension is designed (if i may use the word liberally) with certain notions fixed firmly in mind. chief among them is that the actual C-level calls will be thoroughly ensconced in a suffocating buffer of template hackery, which will employ dark arts and crafts to generate gobs and gobs of impenetrable C at compile time. in fact, GDExtension is so designed that it is basically impossible to use it directly without a code generation layer. unwilling as i was to use C++ (it's against my religion), i wrote my layer from scratch, in janet.
this was not as bad as it could have been. janet has something [^deficient magnificent] in its stdlib that every language should have: a PEG module. the resulting program, ["tool/class-compile.janet] is under 1k lines! just barely. this tool is fired off by the makefile for the ["src/*.gcd] files, generating a header file ["gen/*.h] and an object file ["out/*.o] that implements the class described in the file. the care and feeding of this file format is described in the [>gcd GCD section].
deficient: it is deficient in one particular way: it only operates over bytestrings. so you can use a PEG to parse raw text, or you can use it to implement a lexer, but you can't have both a PEG lexer and parser, which is really fucking dumb and makes things like dealing with whitespace far more painful than it needs to be.)
##gcd GCD language
GCD is a simple IDL which is translated into [^complex much more complicated C]. it's designed to make writing GDExtension classes as close to the native GDScript experience as possible, without the syntactic hanging offenses. you define the structure of the class using GCD, and write inline C to implement your functions. linemarkers are emitted properly so when you inevitably fuck it up, the compiler will be able to apportion the blame properly.
complex: the generated implementation code is roughly 24x longer than the input file
the C "lexing" is pretty robust, and as long as you aren't doing horrible shit with the preprocessor (on the order of ["#define fi }]) it shouldn't choke on anything you write. if you [!are] doing horrible shit with the preprocessor however, [!you deserve everything you get].
|
|
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
>
|
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
..
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
|
%toc @civ { to meet the bare minimum standards necessary to qualify as [!civilized] to my mind, a language must: * support compilation, whether to bytecode or machine code (excludes gdscript) * parse into an AST (excludes bash, php (or used to)) * support AST-rewriting macros (excludes everything outside the lisp family except nim and maybe rust) * use real syntax, not whitespace-based kludges (excludes python, gdscript inter alia) * provide a proper module system (excludes C) * provide for hygienic namespacing (excludes C, gdscript) * provide simple, cheap aggregate types (excludes gdscript, java, maybe python depending on how you view tuples) so, janet and fennel are about the only game in town. } ................................................................................ abi: okay yes in reality it absolutely fucking does, everyone uses the same gods-damned amd one, in actual materially existing reality there is no interop problem at least on x86, but because we haven't ritually enshrined this model at ISO and pacified its marauding [!aramitama] with burnt offerings of motor oil and obsolete DRAM chips, we cover our eyes and pretend we cannot see what is sitting right in front of us. i love too compute dont u so GDExtension is designed (if i may use the word liberally) with certain notions fixed firmly in mind. chief among them is that the actual C-level calls will be thoroughly ensconced in a suffocating buffer of template hackery, which will employ dark arts and crafts to generate gobs and gobs of impenetrable C at compile time. in fact, GDExtension is so designed that it is basically impossible to use it directly without a code generation layer. unwilling as i was to use C++ (it's against my religion), i wrote my layer from scratch, in janet. this was not as bad as it could have been. janet has something [^deficient magnificent] in its stdlib that every language should have: a PEG module. the resulting program, ["tool/class-compile.janet] is under 1k lines! just barely. this tool is fired off by the makefile for the ["src/*.gcd] files, generating a header file ["gen/*.h] and an object file ["out/*.o] that implements the class described in the file. the care and feeding of this file format is described in the [>gcd GCD section]. deficient: it is deficient in one particular way: it only operates over bytestrings. so you can use a PEG to parse raw text, or you can use it to implement a lexer, but you can't have both a PEG lexer and parser, which is really fucking dumb and makes things like dealing with whitespace far more painful than it needs to be.) ##obj object model godot uses a strongly javalike OOP system, which translates poorly to Janet's prototype-based OO. gdjn therefore uses the following convention to translate between the two. the engine executes the "compile-time" phase of a janet script once it is finished loading. it is the responsibility of the compile-time thunk to set up the local environment to represent a class using def statements. the abstract serves as a proxy for method calls. when a lookup is performed against a class proxy, the proxy inspects its attached environment and returns an appropriate object. unadorned immutable bindings are treated as constants. values with the [":field type] annotation are treated as object fields. when a new instance is created, space is allocated for a field of the appropriate type, and initialized to the value of the binding. public [`var] mutable bindins are treated as static variables. ~~~[janet] example bindings (def name "Lisuan") # constant string (def name :field "Lisuan") # field of type string, initializes to "Lisuan" (def name {:field :variant} "Lisuan") field of type variant (def- secret "swordfish") # private constant (def- secret :field "swordfish") # private string field (var count 0) # static variable of type int (var- count 0) # private static variable of type int ~~~ unadorned functions are treated as static functions. private functions (those declared with [`def-]) are available only within the class implementation. they are not exported as methods. functions with the annotation [":method] are treated as methods. when invoked, they are passed a [`self]-reference as their first argument. function type signatures can be specified with the annotations [":takes [...]] and [`:gives [$type]]. tables with the annotation [":subclass] are treated as environment tables specifying inner classes. the macro [`subclass] should generally be used to maintain uniform syntax between outer and inner classes, e.g. ~~~[janet] (use core) (use-classes Object RefCounted) (declare outerClass :is Object) (def val 10) (prop v vec3) (defclass innerClass :is RefCounted (def val 2)) # equivalent to (import prim) (def Object (prim/load-class :Object)) (def RefCounted (prim/load-class :RefCounted)) (def *class-name* :meta :outerClass) (def *class-inherit* :meta Object) (def val 10) (def innerClass (do (def *class-inherit* :meta RefCounted) (let [env (table/setproto @{} (curenv))] (eval '(do (def val 2)) env) env))) ~~~ since the annotations are somewhat verbose, macros are provided to automate the process. + janet + gdscript | ["(def count 10)] | ["const count := 10] | ["(def count {:field int} 10)] | ["var count: int = 10] | ["(defn open [path] ...)] | ["static func open(path: Variant) ...] | ["(defn open {:takes [:string] :gives :int} [path] ...)] | ["static func open(path: String) -> int: ...] | ["(defn close :method [me] ...)] | func close() -> void: ... ##gcd GCD language GCD is a simple IDL which is translated into [^complex much more complicated C]. it's designed to make writing GDExtension classes as close to the native GDScript experience as possible, without the syntactic hanging offenses. you define the structure of the class using GCD, and write inline C to implement your functions. linemarkers are emitted properly so when you inevitably fuck it up, the compiler will be able to apportion the blame properly. complex: the generated implementation code is roughly 24x longer than the input file the C "lexing" is pretty robust, and as long as you aren't doing horrible shit with the preprocessor (on the order of ["#define fi }]) it shouldn't choke on anything you write. if you [!are] doing horrible shit with the preprocessor however, [!you deserve everything you get]. |
Added lib/core.janet version [a791260fb2].
> > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 |
# [ʞ] lib/core.janet # ~ lexi hale <lexi@hale.su> # 🄯 AGPLv3 # ? provides a nicer interface to gdprims # > (import :/lib/core) (import /lib/prim) # this parameter controls the class to inherit from # when a superclass is not explicitly specified. # this should be set to point at RefCounted (def *class-base* (gensym)) (def *class-path* (gensym)) (def- valid-class-flags (do (def list [:abstract :final]) (tabseq [f :in list] f true))) (defn- assemble-meta [args] (defn recurse [acc dox t & r] (cond (string? t) (recurse acc [;dox t] ;r) (or (struct? t) (keyword? t)) (recurse [;acc t] dox ;r) [[;(if (empty? dox) [] [{:doc (string/join dox)}]) ;acc] t ;r])) (recurse [] [] ;args)) (defmacro class [name & dfn] (def def-map (tuple tuple (dyn *current-file*) ;(tuple/sourcemap (dyn :macro-form)))) (defn mapped-val [val] @{:value val :source-map def-map}) (var spec dfn) (var super (dyn *class-base*)) (var export-name (string name)) # documentation is stored differently for classes, # as godot needs to be able to retrieve the text # from a reference to @scope (def dox @[]) (def mode @{}) (forever (match spec [:is super_ & r] # set superclass (do (set super (eval super_)) (set spec r)) [:as ident & r] (do (set export-name (string ident)) (set spec r)) ([m & r] (has-key? valid-class-flags m)) (do (put mode m true) (set spec r)) ([x & r] (string? x)) # docstring (do (array/push dox x) (set spec r)) _ (break))) (def doc-str (string/join dox)) (def body spec) (with-dyns [*class-path* [;(or (dyn *class-path*) []) name]] (with-syms [$local-env $parent-env $form] ~(let [,$parent-env (or @scope (curenv)) ,$local-env (,table/setproto @{ '@name ,(mapped-val (keyword name)) '@gd-id ,(mapped-val (string export-name)) '@inherit ,(mapped-val super) '@mode ,(mapped-val mode) '@doc ,(if (empty? dox) nil (mapped-val doc-str))} ,$parent-env) ,name ,$local-env] (,put ,$local-env '@scope @{:value ,$local-env}) # *self* can be used to access statics (put ,$local-env '@self @{:value (fn @self[& x] (get-in ,$local-env [;x :value]))}) (each ,$form ',body (,eval ,$form ,$local-env)) ,$local-env)))) (defn- member [& x] (string/join [;(or (dyn *class-path*) []) ;x] ".")) (defmacro defclass* [id meta & r] ~(def ,id ,;meta (class ,id ,;r))) (defmacro defclass [id & r] ~(defclass* ,id [] ,;r)) (defmacro defclass- [id & r] ~(defclass* ,id [:private] ,;r)) (defn- parse-sig [sig] (match sig ([ty id dox & r] (string? dox)) [{:type (eval ty) :id id :doc dox} ;(parse-sig r)] [ty id & r] [{:type (eval ty) :id id} ;(parse-sig r) ] _ [])) (defmacro defm [id & rest] (def (meta ret sig & body) (assemble-meta rest)) (def args (parse-sig sig)) (def md [;meta :method {:sig args :ret (eval ret)}]) (def arg-names ['me ;(map (fn [{:id id}] id) args)]) (def path (symbol (member id))) ~(def ,id ,;md (fn ,path ,arg-names ,;body))) (defmacro defm- [id & r] ~(defm ,id :private ,;r)) (defmacro prop [id & args] (def (meta-list ty init-val) (assemble-meta args)) (assert init-val "property missing initial value") ~(def ,id :prop {:type ,ty} ,;meta-list ,init-val)) (defmacro const [id & args] (def (meta-list ty init-val) (assemble-meta args)) (assert init-val "constant missing value") ~(def ,id :const {:type ,ty} ,;meta-list ,init-val)) (defmacro hoist [ids & body] ~(upscope (def ,ids (do ,;body ,ids)))) |
Added lib/prim.janet version [36c3aeafd8].
> > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 |
# [ʞ] lib/prim.janet # ~ lexi hale <lexi@hale.su> # 🄯 AGPLv3 # ? declares the primitives supplied by gdjn # > (import /lib/prim) (def *gd-api-map* @{}) (defmacro- extern [name] (def sym (gensym)) (put *gd-api-map* sym name) ~(def ,name ',sym)) # takes a string and returns a prim/class-handle # for a native godot class (extern class-load) # an abstract that wraps a class and provides # a uniform interface for both native godot # classes and janet environment-classes (extern class) (extern class?) # @-prefixed methods/props are implemented by the vm. # everything else is passed through verbatim. # :@janet? - returns true if underlying class is # a janet env # :@methods - returns a list of methods # {:id method-id :args [(type/variant :Vector3) etc] # :ret (type/variant :String)} # :@super - returns a handle to the superclass # :@new - invokes a constructor, returning an object # use :@janet? and :@env methods to retrieve a # janet environment from a wrapped janet class (extern object) (extern object?) (extern object/class) # method calls on an object are translated to # godot method invocations. index into a class # to retrieve a property. # explicitly create godot values (extern type/dict) (extern type/array) (extern type/array-packed) |
Modified makefile from [ab479c70d3] to [3ece26026f].
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
..
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
|
# form. # # * ["ext] is a Keter-class high-security containment unit for # Other People's Code. # # * ["lib] contains (mostly janet) code that will be included # as blobs in the binary. janet code will be able to import # them. these will be compiled down to .jimage files and # then bundled into rsrc.o. libraries used in ["tool/*] # should only be placed in this directory if they are also # used at runtime in the godot environment (e.g. the OOP # macros). library code used only by tools belongs in the # tool directory. # # * ["out] contains all live binaries and object files. godot = godot4 godot.flags = --headless godot.cmd = "$(godot)" $(godot.flags) janet = $o/janet git = git git.flags.clone = --depth=1 janet.src.path = $x/janet janet.src.git = https://github.com/janet-lang/janet.git janet.root = $x/janet/src janet.cfg = cc.link = -flto cc.comp = -fPIC ifeq ($(debug),1) cc.link += -g cc.comp += -g -O0 endif cc.gdjn.comp = $(cc.comp) \ -std=gnu23 \ -I"$g" \ -I"$s" \ -I"$(janet.root)/include" \ ................................................................................ .PHONY: all clean purge all: $o/gdjn.so clean: rm "$o/"*.o "$g/"*.{jimage,h} "$o/gdjn.so" purge: rm "$o/"* "$g/"* %/:; $(path-ensure) %: | $(@D)/ tags: . find "$s" "$g" -name "*.h" -o -name "*.c" | xargs ctags $o/gdjn.so: $o/gdjn.o $o/rsrc.o $o/interface.o \ $o/janet-lang.o $o/janet-rsrc.o \ $o/libjanet.a "$(cc)" $(cc.gdjn.link) $^ -o"$@" $o/interface.o: $t/c-bind-gen.janet \ $g/interface.h \ $(gd_api_spec) \ $(gd_api_iface) "$(janet)" "$<" loader | "$(cc)" $(cc.gdjn.comp) -c -xc - -o "$@" $g/interface.h: $t/c-bind-gen.janet \ $(gd_api_spec) "$(janet)" "$<" header >"$@" $g/%.h: $s/%.gcd $t/class-compile.janet $(realpath $(janet)) "$(janet)" "$t/class-compile.janet" "$<" header >"$@" $o/%.o: $s/%.gcd $g/%.h $t/class-compile.janet $(realpath $(janet)) "$(janet)" "$t/class-compile.janet" "$<" loader \ | "$(cc)" $(cc.gdjn.comp) -c -xc - -o "$@" $o/%.o: $s/%.c $s/%.h $(realpath $(janet.root)/include/janet.h) "$(cc)" -c $(cc.gdjn.comp) "$<" -o"$@" $o/rsrc.o: $t/rsrc.janet $(realpath $(janet)) \ $g/api.jimage "$(janet)" "$<" -- "$g/api.jimage" %.jimage: $(realpath $(janet)) $g/api.jimage: $t/api-compile.janet $(gd_api_spec) "$(janet)" "$<" "$(gd_api_spec)" "$@" $g/%.jimage: $l/%.janet "$(janet)" -c "$<" "$@" $x/janet $x/janet/src/include/janet.h: "$(git)" $(git.flags) clone $(git.flags.clone) "$(janet.src.git)" "$x/janet" canon = $(realpath $(dir $1))/$(notdir $1) define janet.build = "$(MAKE)" -C "$(janet.src.path)" "$(call canon,$@)" \ JANET_$1="$(call canon,$@)" \ $(janet.cfg) endef $o/libjanet.a: $(janet.src.path) $(call janet.build,STATIC_LIBRARY) $o/janet: $(janet.src.path) $(call janet.build,TARGET) $g/extension_api.json $g/gdextension_interface.h: cd "$g" && $(godot.cmd) --dump-extension-api-with-docs \ --dump-gdextension-interface # individual dependencies janet-header = $(janet.root)/include/janet.h $o/gdjn.o: $s/util.h $g/interface.h $g/janet-lang.h $g/janet-rsrc.h $(gd_api_iface) $o/janet-lang.o: $s/util.h $(janet-header) $o/janet-rsrc.o: $s/util.h $g/janet-lang.h $(janet-header) |
|
|
|
|
|
>
>
>
|
|
|
>
>
>
>
>
>
>
>
>
>
>
|
|
>
|
>
|
<
>
|
|
|
|
|
<
<
<
>
|
|
|
|
|
>
|
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
..
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
|
# form. # # * ["ext] is a Keter-class high-security containment unit for # Other People's Code. # # * ["lib] contains (mostly janet) code that will be included # as blobs in the binary. janet code will be able to import # them. these will be compiled into the api.jimage file and # then bundled into rsrc.o. libraries used in ["tool/*] # should only be placed in ["lib] if they are also used at # runtime in the godot environment (e.g. the OOP macros). # library code used only by tools belongs in the tool # directory. # # * ["out] contains all live binaries and object files. godot = godot4 godot.flags = --headless godot.cmd = "$(godot)" $(godot.flags) janet = $o/janet git = git git.flags.clone = --depth=1 janet.src.path = $x/janet janet.src.git = https://github.com/janet-lang/janet.git janet.root = $x/janet/src janet.cfgfile = $s/janetconf.h janet.cfg = JANETCONF_HEADER=$(realpath janet.cfgfile) janet.cc.bin = -D_gdjn_shims janet.cc.lib = cc.link = -flto cc.comp = -fPIC ifeq ($(debug),1) cc.link += -g cc.comp += -g -O0 -Wall -D_gdjn_build_debug ifndef feat.editor feat.editor = 1 endif else cc.comp += -D_gdjn_build_release endif # are we building with editor support? ifeq ($(feat.editor),1) cc.comp += -D_gdjn_build_editor endif cc.gdjn.comp = $(cc.comp) \ -std=gnu23 \ -I"$g" \ -I"$s" \ -I"$(janet.root)/include" \ ................................................................................ .PHONY: all clean purge all: $o/gdjn.so clean: rm "$o/"*.o "$g/"*.{jimage,h} "$o/gdjn.so" purge: rm "$o/"* "$g/"* $(MAKE) -C "$(janet.src.path)" clean %/:; $(path-ensure) %: | $(@D)/ tags: . find "$s" "$g" -name "*.h" -o -name "*.c" | xargs ctags $o/gdjn.so: $o/gdjn.o $o/rsrc.o $o/interface.o $o/vm.o \ $o/janet-lang.o $o/janet-rsrc.o \ $o/libjanet.a "$(cc)" $(cc.gdjn.link) $^ -o"$@" $o/interface.o: $t/c-bind-gen.janet \ $g/interface.h \ $(gd_api_spec) \ $(gd_api_iface) \ $(janet) "$(janet)" "$<" loader | "$(cc)" $(cc.gdjn.comp) -c -xc - -o "$@" $g/interface.h: $t/c-bind-gen.janet \ $(gd_api_spec) \ $(janet) "$(janet)" "$<" header >"$@" $g/%.h: $s/%.gcd $t/class-compile.janet $(janet) "$(janet)" "$t/class-compile.janet" "$<" header >"$@" $o/%.o: $s/%.gcd $g/%.h $s/gdjn.h $g/interface.h $t/class-compile.janet $(janet) "$(janet)" "$t/class-compile.janet" "$<" loader \ | "$(cc)" $(cc.gdjn.comp) -c -xc - -o "$@" $o/%.o: $s/%.c $s/%.h $(janet.root)/include/janet.h "$(cc)" -c $(cc.gdjn.comp) "$<" -o"$@" $o/rsrc.o: $t/rsrc.janet $(janet) \ $g/api.jimage "$(janet)" "$<" -- "$g/api.jimage" %.jimage: $(janet) $g/api.jimage: $t/api-compile.janet $(wildcard $l/*.janet) $(janet) "$(janet)" "$<" "$@" $x/janet $x/janet/src/include/janet.h: "$(git)" $(git.flags) clone $(git.flags.clone) "$(janet.src.git)" "$x/janet" canon = $(realpath $(dir $1))/$(notdir $1) define janet.build = "$(MAKE)" -C "$(janet.src.path)" "$(call canon,$@)" \ JANET_$1="$(call canon,$@)" \ CFLAGS+="-O2 -g $(janet.cc) $(janet.cc.$2)" \ $(janet.cfg) endef $o/libjanet.a: $(janet.src.path) $(call janet.build,STATIC_LIBRARY,lib) $o/janet: $(janet.src.path) $(call janet.build,TARGET,bin) $g/extension_api.json $g/gdextension_interface.h: cd "$g" && $(godot.cmd) --dump-extension-api-with-docs \ --dump-gdextension-interface # individual dependencies janet-header = $(janet.root)/include/janet.h $o/gdjn.o: $s/util-gd.h $s/type.h $g/interface.h $g/janet-lang.h $g/janet-rsrc.h $(gd_api_iface) $o/janet-lang.o $g/janet-lang.h: $s/util-gd.h $s/type.h $s/util-jn.h $(janet-header) $o/janet-rsrc.o $g/janet-rsrc.h: $s/util-gd.h $s/type.h $s/util-jn.h $g/janet-lang.h $(janet-header) $o/vm.o: $s/util-jn.h |
Modified src/gdjn.c from [f8bd0edc4d] to [65eb39af07].
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 .. 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 .. 75 76 77 78 79 80 81 82 83 84 85 86 87 88 .. 99 100 101 102 103 104 105 106 107 108 109 110 111 112 |
* 🄯 AGPLv3 * ? gdjn entry point */ #include "gdjn.h" #include "janet-lang.h" #include "janet-rsrc.h" gdjn* gdjn_ctx = nullptr; static void gdjn_init ( void* data, GDExtensionInitializationLevel lvl ) { if (lvl != GDEXTENSION_INITIALIZATION_SCENE) return; gdjn_types_fetch(&gdjn_ctx -> gd.t, gdjn_ctx -> gd.getProc); const gdjn_typeDB* c = &gdjn_ctx -> gd.t; gdjn_unit_janetLang_load(); gdjn_unit_janetRsrc_load(); gdjn_ctx -> gd.janetLang_inst = gdjn_class_JanetLang_new() -> self; auto e = gd_engine_registerScriptLanguage( c -> objects.engine, ................................................................................ false ); gd_resourceSaver_addResourceFormatSaver( gdjn_ctx -> gd.t.objects.resourceSaver, gdjn_ctx -> gd.janetSaver_inst, false ); /* gd_variant ret; c -> gd_object.methodBindPtrcall( c -> gd_m_engine.registerScriptLanguage_ptr, c -> objects.engine, (GDExtensionConstTypePtr[]) { &gdjn_ctx -> gd.janetLang_inst, }, &ret ); */ _t(array).empty(&gdjn_ctx -> gd.dox, nullptr); gd_stringName empty = {}; _t(stringName).empty(&empty, nullptr); _t(array).setTyped(&gdjn_ctx -> gd.dox, GDEXTENSION_VARIANT_TYPE_DICTIONARY, &empty, &empty); ................................................................................ ( void* data, GDExtensionInitializationLevel lvl ) { if (lvl != GDEXTENSION_INITIALIZATION_SCENE) return; /* we get double frees otherwise */ const gdjn_typeDB* c = &gdjn_ctx -> gd.t; gd_engine_unregisterScriptLanguage( c -> objects.engine, gdjn_ctx -> gd.janetLang_inst ); /* GDExtensionTypePtr ret; ................................................................................ gd_resourceSaver_removeResourceFormatSaver( gdjn_ctx -> gd.t.objects.resourceSaver, gdjn_ctx -> gd.janetSaver_inst ); /* gd_refCounted_unreference(gdjn_ctx -> gd.janetLoader_inst); */ /* gd_refCounted_unreference(gdjn_ctx -> gd.janetSaver_inst); */ gdjn_class_JanetLang_del(nullptr, gdjn_ctx -> gd.janetLang_inst); gdjn_ctx -> gd.free(gdjn_ctx); gdjn_ctx = nullptr; } gdBool |
> > > > > > > > > > > > > > > > < < < < < < < < < < > > > |
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 .. 62 63 64 65 66 67 68 69 70 71 72 73 74 75 .. 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 ... 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 |
* 🄯 AGPLv3 * ? gdjn entry point */ #include "gdjn.h" #include "janet-lang.h" #include "janet-rsrc.h" #include "vm.h" gdjn* gdjn_ctx = nullptr; void* gdjn_janet_malloc (size_t sz) { return _alloc(char, sz); } void gdjn_janet_free (void* ptr) { _free(ptr); } void* gdjn_janet_realloc(void* ptr, size_t sz) { return _realloc(ptr, char, sz); } void* gdjn_janet_calloc (size_t n, size_t esz) { const size_t sz = esz*n; void* v = _alloc(char, sz); memset(v, 0, sz); return v; } static void gdjn_init ( void* data, GDExtensionInitializationLevel lvl ) { if (lvl != GDEXTENSION_INITIALIZATION_SCENE) return; gdjn_types_fetch(&gdjn_ctx -> gd.t, gdjn_ctx -> gd.getProc); const gdjn_typeDB* c = &gdjn_ctx -> gd.t; janet_init(); gdjn_ctx -> jn.api = gdjn_vm_api_build_core(); janet_gcroot(janet_wrap_table(gdjn_ctx -> jn.api)); gdjn_unit_janetLang_load(); gdjn_unit_janetRsrc_load(); gdjn_ctx -> gd.janetLang_inst = gdjn_class_JanetLang_new() -> self; auto e = gd_engine_registerScriptLanguage( c -> objects.engine, ................................................................................ false ); gd_resourceSaver_addResourceFormatSaver( gdjn_ctx -> gd.t.objects.resourceSaver, gdjn_ctx -> gd.janetSaver_inst, false ); _t(array).empty(&gdjn_ctx -> gd.dox, nullptr); gd_stringName empty = {}; _t(stringName).empty(&empty, nullptr); _t(array).setTyped(&gdjn_ctx -> gd.dox, GDEXTENSION_VARIANT_TYPE_DICTIONARY, &empty, &empty); ................................................................................ ( void* data, GDExtensionInitializationLevel lvl ) { if (lvl != GDEXTENSION_INITIALIZATION_SCENE) return; /* we get double frees otherwise */ const gdjn_typeDB* c = &gdjn_ctx -> gd.t; janet_gcunroot(janet_wrap_table(gdjn_ctx -> jn.api)); gd_engine_unregisterScriptLanguage( c -> objects.engine, gdjn_ctx -> gd.janetLang_inst ); /* GDExtensionTypePtr ret; ................................................................................ gd_resourceSaver_removeResourceFormatSaver( gdjn_ctx -> gd.t.objects.resourceSaver, gdjn_ctx -> gd.janetSaver_inst ); /* gd_refCounted_unreference(gdjn_ctx -> gd.janetLoader_inst); */ /* gd_refCounted_unreference(gdjn_ctx -> gd.janetSaver_inst); */ gdjn_class_JanetLang_del(nullptr, gdjn_ctx -> gd.janetLang_inst); janet_deinit(); gdjn_ctx -> gd.free(gdjn_ctx); gdjn_ctx = nullptr; } gdBool |
Modified src/gdjn.h from [b22d9ec405] to [a8262d25e1].
17
18
19
20
21
22
23
24
25
26
27
28
29
30
..
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
|
#define _warn(msg) _emit(warn, msg) #define _err(msg) _emit(err, msg) typedef GDExtensionBool gdBool; #define _alloc(ty, n) \ ((typeof(ty)*)gdjn_alloc(sizeof(ty) * (n))) #define _free(v) \ (gdjn_ctx -> gd.free(v)) #define _sz(r) ((sizeof(r) / sizeof(*r))) #define _t(T) \ (gdjn_ctx -> gd.t.gd_##T) #define _method(name) \ ................................................................................ GDExtensionObjectPtr janetLang_inst, janetLoader_inst, janetSaver_inst; } gd; struct gdjn_jn { Janet env; } jn; } gdjn; extern gdjn* gdjn_ctx; [[gnu::alloc_size(1)]] static inline void* gdjn_alloc(size_t sz) { return gdjn_ctx -> gd.alloc(sz); } typedef struct gdjn_gd gdjn_gd; // derp typedef struct gdjn_jn gdjn_jn; typedef struct gdjn_class_def gdjn_class_def; void gdjn_dox ( gd_dictionary* page ); |
>
>
|
>
>
>
>
|
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
..
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
|
#define _warn(msg) _emit(warn, msg) #define _err(msg) _emit(err, msg) typedef GDExtensionBool gdBool; #define _alloc(ty, n) \ ((typeof(ty)*)gdjn_alloc(sizeof(ty) * (n))) #define _realloc(v, ty, n) \ ((typeof(ty)*)gdjn_realloc((v), sizeof(ty) * (n))) #define _free(v) \ (gdjn_ctx -> gd.free(v)) #define _sz(r) ((sizeof(r) / sizeof(*r))) #define _t(T) \ (gdjn_ctx -> gd.t.gd_##T) #define _method(name) \ ................................................................................ GDExtensionObjectPtr janetLang_inst, janetLoader_inst, janetSaver_inst; } gd; struct gdjn_jn { JanetTable* api; } jn; } gdjn; extern gdjn* gdjn_ctx; [[gnu::alloc_size(1)]] static inline void* gdjn_alloc(size_t sz) { return gdjn_ctx -> gd.alloc(sz); } [[gnu::alloc_size(2)]] static inline void* gdjn_realloc(void* v, size_t sz) { return gdjn_ctx -> gd.realloc(v, sz); } typedef struct gdjn_gd gdjn_gd; // derp typedef struct gdjn_jn gdjn_jn; typedef struct gdjn_class_def gdjn_class_def; void gdjn_dox ( gd_dictionary* page ); |
Modified src/janet-lang.gcd from [c1b1ca0dcf] to [101fec1406].
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 .. 87 88 89 90 91 92 93 94 95 96 97 98 99 100 ... 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 ... 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 ... 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 |
(* [ʞ] src/janet-lang.gcd vi:ft=d * ~ lexi hale <lexi@hale.su> * 🄯 AGPLv3 * ? implement the godot-janet interface *) use <janet.h>; class JanetLang is ScriptLanguageExtension { use <stdio.h>; use "util.h"; new {}; impl _get_name() -> string { gd_string j; _t(string).newWithUtf8Chars(&j, "Janet"); return j; }; ................................................................................ l("for"), l("forv"), l("forever"), l("seq"), l("catseq"), l("map"), l("mapcat"), l("find"), l("array"), l("tuple"), l("string"), l("buffer"), l("table"), l("struct"), (* et cetera ad nauseam *) }; gd_packedStringArray r = {}; _t(packedStringArray).empty(&r, nullptr); for (size_t i = 0; i < _sz(words); ++i) { gdu_array_string_pushPtr(&r, words[i].w, words[i].sz); } ................................................................................ return janscr -> self; }; impl _create_script() -> ref Object { auto janscr = gdjn_class_JanetScriptText_new(); return janscr -> self; }; impl _get_documentation() -> array[dictionary] { gd_array a = {}; _t(array).ref(&a, &gdjn_ctx -> gd.dox); return gdjn_ctx -> gd.dox; }; impl _init() { /* (* "deprecated" but i still have to impl it?? *) */ }; impl _frame() {}; impl _thread_enter() { janet_init(); }; impl _thread_exit() { janet_deinit(); }; impl _finish() {}; ................................................................................ _t(dictionary).empty(&dict,nullptr); return dict; }; }; class JanetScript is ScriptExtension { var as string: path; new { _t(string).empty(&me -> path, nullptr); }; del { _t(string).dtor(&me -> path); }; impl _create_instance(array[variant] argv, int argc, ref Object owner, bool refCounted, int error) -> ref Object { }; impl _get_language() -> ref ScriptLanguage { return gdjn_ctx -> gd.janetLang_inst; }; impl _set_path(string path, bool takeOver) { if (takeOver) { _t(string).dtor(&me -> path); me -> path = path; } else { _t(string).dtor(&me -> path); /* _t(string).copy(&me -> path, (void const*[]) {&path}); */ me -> path = gdu_string_dup(&path); } }; impl _get_base_script() -> ref Script { return nullptr; }; impl _has_static_method(string-name method) -> bool { return false; }; ................................................................................ impl _get_global_name(string-name id) -> string-name { return gdu_sym_null(); }; impl _is_abstract() -> bool { return false; (* TODO abuse for non-class scripts? *) }; impl _get_instance_base_type() -> string-name { return _gdu_intern("Object"); }; impl _is_valid() -> bool { return true; }; impl _get_documentation() -> array[dictionary] { gd_array dox; _t(array).empty(&dox, nullptr); auto empty = gdu_sym_null(); _t(array).setTyped(&dox, GDEXTENSION_VARIANT_TYPE_DICTIONARY, &empty, &empty); _t(stringName).dtor(&empty); return dox; }; }; class JanetScriptText extends JanetScript { var as string: src; new { _t(string).empty(&me -> src, nullptr); }; del { _t(string).dtor(&me -> src); }; impl _has_source_code() -> bool { return true; }; impl _get_source_code() -> string { auto d = gdu_string_dup(&me -> src); return d; }; impl _set_source_code(string s) { _t(string).dtor(&me -> src); _t(string).copy(&me -> src, (void const*[]) {&s}); }; impl _reload(bool keepState) -> int (* Error *) { (* TODO *) return gd_Error_ok; }; }; import { typedef struct gdjn_janet_image { size_t sz; uint8_t* buf; |
> > > | | < > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > < > > > > > > > > > > > > > > > > > > > > > > > > > > > > > < | < < < | | < | > > > > > > > > > > > > > > | | | > | | > > > > | | < | > > > > > > > > > > > > > > > > > > > > > > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 .. 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 ... 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 ... 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 ... 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 |
(* [ʞ] src/janet-lang.gcd vi:ft=d * ~ lexi hale <lexi@hale.su> * 🄯 AGPLv3 * ? implement the godot-janet interface *) import "type.h"; import "util-gd.h"; use "util-jn.h"; use <janet.h>; class JanetLang is ScriptLanguageExtension { (* use <stdio.h>; *) new {}; (* var as array[ref Object]: placeholders; *) impl _get_name() -> string { gd_string j; _t(string).newWithUtf8Chars(&j, "Janet"); return j; }; ................................................................................ l("for"), l("forv"), l("forever"), l("seq"), l("catseq"), l("map"), l("mapcat"), l("find"), l("array"), l("tuple"), l("string"), l("buffer"), l("table"), l("struct"), l("print"), l("prin"), l("pp"), (* gdjn macros *) l("defm"), l("defm-"), l("class"), l("defclass"), l("defclass-"), l("prop"), (* et cetera ad nauseam *) }; gd_packedStringArray r = {}; _t(packedStringArray).empty(&r, nullptr); for (size_t i = 0; i < _sz(words); ++i) { gdu_array_string_pushPtr(&r, words[i].w, words[i].sz); } ................................................................................ return janscr -> self; }; impl _create_script() -> ref Object { auto janscr = gdjn_class_JanetScriptText_new(); return janscr -> self; }; use { #define _typeCode(x) gd_ScriptLanguageExtension_LookupResultType_lookupResult##x }; impl _get_documentation() -> array[dictionary] { gd_array a = {}; _t(array).ref(&a, &gdjn_ctx -> gd.dox); return gdjn_ctx -> gd.dox; }; impl _lookup_code ( string code; string symbol; string path; ref Object owner; ) -> dictionary { gd_dictionary d; _t(dictionary).empty(&d, nullptr); gd_variant v; v = gd_variant_of_int(gd_Error_failed); gdu_setKey(&d, _pstrLit("result"), &v); v = gd_variant_of_int(_typeCode(ScriptLocation)); gdu_setKey(&d, _pstrLit("type"), &v); return d; }; impl _init() { /* (* "deprecated" but i still have to impl it?? *) */ }; impl _frame() {}; impl _thread_enter() { janet_init(); }; impl _thread_exit() { janet_deinit(); }; impl _finish() {}; ................................................................................ _t(dictionary).empty(&dict,nullptr); return dict; }; }; class JanetScript is ScriptExtension { var as string: path; var JanetTable*: env; new { _t(string).empty(&me -> path, nullptr); me -> env = nullptr; }; del { _t(string).dtor(&me -> path); if (me -> env) janet_gcunroot(janet_wrap_table(me -> env)); }; class BaseInstance is Object { var as ref JanetScript: script; del { gd_refCounted_unreference(me -> script); printf("del script instance\n"); }; }; class Instance extends JanetScript_BaseInstance { }; class Placeholder extends JanetScript_BaseInstance { }; (* impl _create_instance(array[variant] argv, int argc, ref Object owner, bool refCounted, int error) -> ref Object { *) impl _can_instantiate() -> bool { if (me -> env == nullptr) return false; return true; // FIXME }; impl _instance_create(ref Object obj) -> ref Object { auto ci = gdjn_class_JanetScript_Instance_new(); printf("made new script instance\n"); ci -> super.script = me; gd_refCounted_reference(me -> self); return ci; }; impl _placeholder_instance_create(ref Object obj) -> ref Object { auto ci = gdjn_class_JanetScript_Placeholder_new(); printf("made new script placeholder\n"); ci -> super.script = me; gd_refCounted_reference(me -> self); return ci; }; impl _get_language() -> ref ScriptLanguage { return gdjn_ctx -> gd.janetLang_inst; }; impl _set_path(string path, bool takeOver) { _t(string).dtor(&me -> path); /* _t(string).copy(&me -> path, (void const*[]) {&path}); */ me -> path = gdu_string_dup(&path); }; impl _get_base_script() -> ref Script { return nullptr; }; impl _has_static_method(string-name method) -> bool { return false; }; ................................................................................ impl _get_global_name(string-name id) -> string-name { return gdu_sym_null(); }; impl _is_abstract() -> bool { return false; (* TODO abuse for non-class scripts? *) }; impl _get_instance_base_type() -> string-name { return _gdu_intern("ScriptExtension"); }; impl _is_valid() -> bool { return true; }; impl _get_documentation() -> array[dictionary] { gd_array dox; _t(array).empty(&dox, nullptr); auto empty = gdu_sym_null(); _t(array).setTyped(&dox, GDEXTENSION_VARIANT_TYPE_DICTIONARY, &empty, &empty); _t(stringName).dtor(&empty); return dox; }; (* impl _update_exports() { }; impl _placeholder_instance_create (ref Object forObj) -> ref Object { }; impl _placeholder_erased (ref Object ph) { }; *) }; class JanetScriptText extends JanetScript { var pstr: src; new { me -> src = (pstr){}; }; del { _drop(me -> src); }; impl _has_source_code() -> bool { return true; }; impl _get_source_code() -> string { return gdu_str_sz(me -> src.v, me -> src.sz); /* auto d = gdu_string_dup(&me -> src); */ /* return d; */ }; impl _set_source_code(string s) { _drop(me -> src); me -> src = gdu_string_pdup(&s); /* printf("janet: set script %d %s\n", (int)me -> src.sz, me -> src.v); */ gdjn_class_JanetScriptText_method__reload(me, false); /* _t(string).dtor(&me -> src); */ /* _t(string).copy(&me -> src, (void const*[]) {&s}); */ }; impl _reload(bool keepState) -> int (* Error *) { auto errorCode = gd_Error_ok; if (me -> super.env) janet_gcunroot(janet_wrap_table(me -> super.env)); me -> super.env = nullptr; pstr path = _gdu_string_stackp(&me -> super.path); auto cls = jnu_table_extend(gdjn_ctx -> jn.api, 8); janet_gcroot(janet_wrap_table(cls)); /* printf("janet: doing bytes %d %s\n", (int)me -> src.sz, me -> src.v); */ int e = janet_dobytes(cls, (uint8_t const*)me -> src.v, me -> src.sz, path.v, nullptr); (* we discard the return value; the environment is what * we're really interested in here *) /* printf("janet: bytes done, got %d\n",e); */ if (e == 0) { me -> super.env = cls; } else { _err("janet module could not be loaded"); errorCode = gd_Error_errParseError; janet_gcunroot(janet_wrap_table(cls)); } return errorCode; }; }; import { typedef struct gdjn_janet_image { size_t sz; uint8_t* buf; |
Modified src/janet-rsrc.gcd from [ab1704aae3] to [2d0266f6b4].
1 2 3 4 5 6 7 8 9 10 11 12 13 .. 40 41 42 43 44 45 46 47 48 49 50 51 52 53 .. 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 .. 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 |
(* [ʞ] src/janet-rsrc.gcd vi:ft=d * ~ lexi hale <lexi@hale.su> * 🄯 AGPLv3 * ? implement the saving and loading of janet scripts *) use "util.h"; use <assert.h>; use "janet-lang.h"; use { static gd_packedStringArray janetExts(void) { gd_packedStringArray r = {}; ................................................................................ || gdu_symEq(&type, "JanetScriptImage"); }; impl _get_resource_type(string path) -> string { const char* str = ""; switch (janetKind(&path)) { case janetFileImage: str="JanetScriptImage"; break; case janetFileText: str="JanetScriptText"; break; } return gdu_str(str); }; use { static inline gd_variant vFromErr(int64_t err) { gd_variant v; ................................................................................ }; impl _load ( string path; string origPath; bool subThreads; int cacheMode; ) -> variant { switch (janetKind(&path)) { case janetFileImage: { auto s = gdjn_class_JanetScriptImage_new(); return vFromObj(gdu_cast(s->self, "Resource")); }; case janetFileText: { auto s = gdjn_class_JanetScriptText_new(); return vFromObj(gdu_cast(s->self, "Resource")); }; default: { return vFromErr(gd_Error_errFileUnrecognized); }; } }; }; class JanetScriptSaver is ResourceFormatSaver { use { static inline bool ................................................................................ || _gdu_objIs(res, JanetScriptText); } }; impl _get_recognized_extensions() -> packed-string-array { return janetExts(); }; impl _recognize(ref Resource res) -> bool { return gdjn_isJanet(res); }; impl _save(ref Resource res, string path, int flags) -> int { gd_refCounted_reference(res); assert(gdjn_isJanet(res)); gd_string path_mine; _t(string).copy(&path_mine, (void const*[]) {&path}); auto fd = gd_fileAccess_open(path, gd_FileAccess_ModeFlags_write); gd_refCounted_reference(fd); if (_gdu_objIs(res, JanetScriptText)) { auto asText = gdu_cast(res, "JanetScriptText"); gd_string src = gd_script_getSourceCode(asText); gd_fileAccess_storeString(fd, src); _t(string).dtor(&src); auto data = gdjn_class_JanetScriptText_data(res); } else if (_gdu_objIs(res, JanetScriptImage)) { auto asImg = gdu_cast(res, "JanetScriptImage"); auto data = gdjn_class_JanetScriptImage_data(res); }; gd_fileAccess_close(fd); _t(string).dtor(&path_mine); gd_refCounted_unreference(fd); gd_refCounted_unreference(res); }; }; |
| > > > > > > > > > > > > | > > > | > > > > > > > | < < > > > > > > | < < | | | > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 .. 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 .. 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 ... 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 |
(* [ʞ] src/janet-rsrc.gcd vi:ft=d * ~ lexi hale <lexi@hale.su> * 🄯 AGPLv3 * ? implement the saving and loading of janet scripts *) use "util-gd.h"; use <assert.h>; use "janet-lang.h"; use { static gd_packedStringArray janetExts(void) { gd_packedStringArray r = {}; ................................................................................ || gdu_symEq(&type, "JanetScriptImage"); }; impl _get_resource_type(string path) -> string { const char* str = ""; switch (janetKind(&path)) { case janetFileImage: str="JanetScriptImage"; break; case janetFileText: str="JanetScriptText"; break; default: break; } return gdu_str(str); }; use { static inline gd_variant vFromErr(int64_t err) { gd_variant v; ................................................................................ }; impl _load ( string path; string origPath; bool subThreads; int cacheMode; ) -> variant { (* yes it's a bit hinky using hardwired static dispatch here * but ye gods, at least it spares us from having to use the * horrible gdscript ptrcall mechanism *) GDExtensionObjectPtr obj = nullptr; auto cpath = _gdu_string_stackp(&path); gd_string path_mine = gdu_str_sz(cpath.v, cpath.sz); auto fd = gd_fileAccess_open(path_mine, gd_FileAccess_ModeFlags_read); // auto cpath = _gdu_string_stackp(&path_mine); printf("janet: loading from file %zu %s\n", cpath.sz, cpath.v); /* gd_refCounted_reference(fd); */ switch (janetKind(&path)) { case janetFileImage: { auto s = gdjn_class_JanetScriptImage_new(); gdjn_class_JanetScript_method__set_path(&s->super, path_mine, false); gdjn_class_JanetScriptImage_method__reload(s, false); obj = s -> self; break; }; case janetFileText: { auto s = gdjn_class_JanetScriptText_new(); gdjn_class_JanetScript_method__set_path(&s->super, path_mine, false); auto text = gd_fileAccess_getAsText(fd, false); gdjn_class_JanetScriptText_method__set_source_code (s, text); _t(string).dtor(&text); obj = s -> self; break; }; default: break; } gd_fileAccess_close(fd); _t(string).dtor(&path_mine); /* gd_refCounted_unreference(fd); */ if (obj) return vFromObj(gdu_cast(obj, "Resource")); else return vFromErr(gd_Error_errFileUnrecognized); }; }; class JanetScriptSaver is ResourceFormatSaver { use { static inline bool ................................................................................ || _gdu_objIs(res, JanetScriptText); } }; impl _get_recognized_extensions() -> packed-string-array { return janetExts(); }; impl _recognize(ref Resource res) -> bool { printf("checking against res %p\n", res); return gdjn_isJanet(res); }; impl _save(ref Resource res, string path, int flags) -> int { gd_refCounted_reference(res); assert(gdjn_isJanet(res)); gd_string path_mine = gdu_string_dup(&path); auto fd = gd_fileAccess_open(path, gd_FileAccess_ModeFlags_write); gd_refCounted_reference(fd); if (_gdu_objIs(res, JanetScriptText)) { auto asText = gdu_cast(res, "JanetScriptText"); gd_string src = gd_script_getSourceCode(asText); gd_fileAccess_storeString(fd, src); _t(string).dtor(&src); } else if (_gdu_objIs(res, JanetScriptImage)) { // auto asImg = gdu_cast(res, "JanetScriptImage"); // auto data = gdjn_class_JanetScriptImage_data(res); }; // gd_fileAccess_close(fd); _t(string).dtor(&path_mine); printf("number of surviving saver refs: %zu\n", gd_refCounted_getReferenceCount(gdu_cast(me -> self, "RefCounted"))); gd_refCounted_unreference(fd); gd_refCounted_unreference(res); return gd_Error_ok; }; }; |
Added src/janetconf.h version [70cb9a69cc].
> > > > > > > > > > > > > > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
#pragma once #ifdef _gdjn_shims // install shims for code compiled in library mode // yes this is awful, why do you ask void* gdjn_janet_malloc(size_t sz) { return malloc(sz); } void* gdjn_janet_realloc(void* ptr, size_t sz) { return realloc(ptr, sz); } void* gdjn_janet_calloc(size_t n, size_t sz) { return calloc(n, sz); } void gdjn_janet_free(void* ptr) { free(ptr); } #else # define janet_malloc gdjn_janet_malloc # define janet_calloc gdjn_janet_calloc # define janet_realloc gdjn_janet_realloc # define janet_free gdjn_janet_free #endif |
Added src/type.h version [7c7116050b].
> > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 |
/* [ʞ] src/type.h * ~ lexi hale <lexi@hale.su> * 🄯 AGPLv3 * ? miscellaneous useful types & macros */ #pragma once #define _strDynSz(a) \ __builtin_choose_expr( \ /* if */ __builtin_types_compatible_p(typeof(a), pstr), \ /* then */ ((pstr*)&(a))->sz, \ /* else */ strlen(*(char const**)&(a))) #define _strDynMem(a) \ __builtin_choose_expr( \ /* if */ __builtin_types_compatible_p(typeof(a), pstr), \ /* then */ ((pstr*)&(a))->v, \ /* else */ *(char**)&(a)) #define _pstrOf(a) ((pstr){.v=_strDynMem(a), .sz=_strDynSz(a)}) #define _array(t) struct {t* v; size_t sz;} typedef _array(char) pstr; #define _pstrLit(a) ((pstr){.v=(a),.sz=sizeof(a)-1}) #define _drop(x) ({ \ if (x.v) { \ _free (x.v); \ x.v = nullptr; \ x.sz = 0; \ } \ }) #define _new(T, n) ((T){ \ .v = _alloc( typeof( ((typeof(T)) {}).v ), n), \ .sz = n, \ }) |
Modified src/util-gd.h from [70ed0e36b0] to [79dbc22466].
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 .. 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 ... 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 ... 371 372 373 374 375 376 377 |
/* [ʞ] util.h * ~ lexi hale <lexi@hale.su> * 🄯 AGPLv3 * ? encapsulate annoying operations (read: pitiful, fragile blast * shield over the most indefensibly psychotic pieces of the godot * "type" "system") * * if you want to use this outside gdjn, redefine the macro _t * from gdjn.h appropriately. * * (honestly tho you should use c-bind-gen.janet too) */ #pragma once #include "gdjn.h" #include <string.h> static inline gd_string gdu_string_of_stringName(const gd_stringName* const s) { gd_string r; _t(string).fromStringName(&r, (void*)&s); return r; } ................................................................................ #define _refMut(x) typeof(typeof(x) * const) #define _with(T, k, v, ...) ({\ typeof(gd_##T) k = v; \ do { __VA_ARGS__; } while (0); \ _t(T).dtor(&k); \ }) #define _withSym(k, v, ...) \ _with(stringName, k, gdu_intern(v), __VA_ARGS__) #define _withSym0(k, ...) \ _with(stringName, k, {}, __VA_ARGS__) #define _withStr(k, v, ...) \ _with(string, k, gdu_str(v), __VA_ARGS__) #define _withStr0(k, v, ...) \ _with(string, k, {}, __VA_ARGS__) #define _typeEq(a, b) \ __builtin_classify_type(typeof(a)) == _builtin_classify_type(typeof(b)) #define _refVal 0 #define _refPtr 1 #define _refArray 2 #define _indirect(a) \ __builtin_types_compatible_p(typeof(a), void*) #define _refKind(a) \ __builtin_choose( _typeEq(typeof_unqual(a), \ typeof_unqual(a[0]) []), _refArray \ /* false */ __builtin_choose(_typeEq(a, (typeof_unqual(a[0]) *)), _refPtr )) #define _szElse(a, zero) \ __builtin_choose(_refKind(a) == _refArray, \ /* true */ _sz(a), \ /* false */ __builtin_choose(_refKind(a) == _refPtr, \ /* true */ (__builtin_counted_by(ptr) != nullptr ? \ *__builtin_counted_by(ptr) : (zero)) \ /* false */ (void)0 /* bad type */ )) #define _sz0(a) _szElse(a,0) #define _szStr(a) \ __builtin_choose(_typeEq((a), pstr), (a).sz, _szElse(a, strlen(a))) #define _array(t) struct {t* v; size_t sz;} typedef _array(char) pstr; #define _strWithSz(a) (a), _strLen(a) static inline bool gdu_symIs ( gd_stringName const* const a, gd_stringName const* const b ) { bool res; ................................................................................ char* buf = _alloc(char, len); gdu_string_emit(s, buf, len - 1); return (pstr){buf,len}; } static inline gd_string gdu_string_dup (_ref(gd_string) s) { gd_string cp; _t(string).copy(&cp, (void const*[]) {s}); return cp; } #define _cat(x,y) x##y #define __cat(x,y) _cat(x,y) #define _gensym __cat(_sym_,__COUNTER__) #define __gdu_string_auto(szid, name, str) \ ................................................................................ static inline GDExtensionObjectPtr gdu_cast ( GDExtensionConstObjectPtr what, _ref(char) to ) { return _t(object).castTo(what, gdu_classTag(to)); } |
| | | | < < < < < > | | | | | | | | | | < > | | | < > | < < < < < < < < < < < < < < < < < < < < < < | > > > > > > | > > > | > > > > > > > > > > > > > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 .. 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 ... 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 ... 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 |
/* [ʞ] src/util-gd.h * ~ lexi hale <lexi@hale.su> * 🄯 AGPLv3 * ? encapsulate annoying godot operations (read: pitiful, * fragile blast shield over the most indefensibly psychotic * pieces of the godot "type" "system") */ #pragma once #include "gdjn.h" #include <string.h> #include "type.h" static inline gd_string gdu_string_of_stringName(const gd_stringName* const s) { gd_string r; _t(string).fromStringName(&r, (void*)&s); return r; } ................................................................................ #define _refMut(x) typeof(typeof(x) * const) #define _with(T, k, v, ...) ({\ typeof(gd_##T) k = v; \ do { __VA_ARGS__; } while (0); \ _t(T).dtor(&k); \ }) #define _withSym0(k, ...) \ _with(stringName, k, {}, __VA_ARGS__) #define _withSymSz(k, v, sz, ...) \ _with(stringName, k, gdu_intern_sz(v, sz), __VA_ARGS__) #define _withSym(k, v, ...) \ _withSymSz(k, _strDynMem(v), _strDynSz(v), __VA_ARGS__) #define _withSymLit(k, v, ...) \ _withSymSz(k, _litSz(v), __VA_ARGS__) #define _withStr0(k, ...) \ _with(string, k, {}, __VA_ARGS__) #define _withStrSz(k, v, sz, ...) \ _with(string, k, gdu_str_sz(v, sz), __VA_ARGS__) #define _withStr(k, v, ...) \ _withStrSz(k, _strDynMem(v), _strDynSz(v), __VA_ARGS__) #define _withStrLit(k, v, ...) \ _withStrSz(k, _litSz(v), __VA_ARGS__) static inline bool gdu_symIs ( gd_stringName const* const a, gd_stringName const* const b ) { bool res; ................................................................................ char* buf = _alloc(char, len); gdu_string_emit(s, buf, len - 1); return (pstr){buf,len}; } static inline gd_string gdu_string_dup (_ref(gd_string) s) { /* the godot copy method seems to be broken somehow, * probably for reasons that have to do with the * hypercomplicated CoW implementation that i think * is meant to be handled on this end somehow? we * can seemingly avoid crashes and memory corruption * if we copy the string through C, forcing a clean * new string to be generated on the godot side. */ // _t(string).copy(&cp, (void const*[]) {s}); auto cstr = gdu_string_pdup(s); auto copied = gdu_str_sz(cstr.v, cstr.sz); _drop(cstr); return copied; } #define _cat(x,y) x##y #define __cat(x,y) _cat(x,y) #define _gensym __cat(_sym_,__COUNTER__) #define __gdu_string_auto(szid, name, str) \ ................................................................................ static inline GDExtensionObjectPtr gdu_cast ( GDExtensionConstObjectPtr what, _ref(char) to ) { return _t(object).castTo(what, gdu_classTag(to)); } static inline void gdu_setKey ( gd_dictionary* const dict, pstr const key, _ref(gd_variant) val ) { gd_variant v = gd_variant_of_dictionary(*dict); _withSym(keyName, key, { uint8_t ok = false; _t(variant).setNamed(&v, &keyName, val, &ok); }); } |
Added src/util-jn.h version [7073b831fd].
> > > > > > > > > > > > > > > > > > > > > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
/* [ʞ] util-gn.h * ~ lexi hale <lexi@hale.su> * 🄯 AGPLv3 * ? convenience functions for janet */ #pragma once #include <janet.h> static inline void jnu_table_inherit(Janet tbl, Janet proto) { auto t = janet_unwrap_table(tbl); auto p = janet_unwrap_table(proto); t -> proto = p; } static inline JanetTable* jnu_table_extend(JanetTable* p, size_t sz) { auto t = janet_table(sz); t -> proto = p; return t; } |
Added src/vm.c version [57a8603486].
> > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 |
#include "vm.h" #include "util-jn.h" #include "util-gd.h" #include "rsrc.h" #define _safe_wrap(gdt, ct) ({ \ _Static_assert(sizeof(gdt) == sizeof(ct)); \ *(typeof(ct)*)v; \ }) void gdjn_dejanetize_typed ( GDExtensionTypePtr v, GDExtensionVariantType const t, Janet val ) { switch (t) { case GDEXTENSION_VARIANT_TYPE_BOOL: assert(janet_type(val) == JANET_BOOLEAN); *(bool*)v = janet_unwrap_boolean(val); break; case GDEXTENSION_VARIANT_TYPE_INT: switch (janet_type(val)) { case JANET_NUMBER: *(int64_t*)v = janet_unwrap_integer(val); break; case JANET_INT_S64: *(int64_t*)v = janet_unwrap_s64(val); break; case JANET_INT_U64: *(int64_t*)v = janet_unwrap_u64(val); break; default: assert(false); } break; case GDEXTENSION_VARIANT_TYPE_STRING: case GDEXTENSION_VARIANT_TYPE_STRING_NAME: { JanetString str; switch (janet_type(val)) { case JANET_STRING: str = janet_unwrap_string(val); break; case JANET_KEYWORD: str = janet_unwrap_keyword(val); break; case JANET_SYMBOL: str = janet_unwrap_symbol(val); break; default: assert(false); } size_t len = janet_string_length(str); if (t == GDEXTENSION_VARIANT_TYPE_STRING_NAME) { _t(stringName).newWithUtf8CharsAndLen(v, (char*)str, len); } else { _t(string).newWithUtf8CharsAndLen(v, (char*)str, len); } break; } case GDEXTENSION_VARIANT_TYPE_ARRAY: { } case GDEXTENSION_VARIANT_TYPE_DICTIONARY: { } default: { assert(false); } } } Janet gdjn_janetize_typed ( GDExtensionTypePtr const v, GDExtensionVariantType const t ) { switch (t) { case GDEXTENSION_VARIANT_TYPE_NIL: return janet_wrap_nil(); case GDEXTENSION_VARIANT_TYPE_BOOL: return janet_wrap_boolean(_safe_wrap(gd_bool, int8_t)); case GDEXTENSION_VARIANT_TYPE_INT: return janet_wrap_s64(_safe_wrap(gd_int, int64_t)); case GDEXTENSION_VARIANT_TYPE_FLOAT: _Static_assert( sizeof(gd_float) == sizeof(double) || sizeof(gd_float) == sizeof(float) ); return janet_wrap_number( (sizeof(gd_float) == sizeof(double)) ? *(double*)v : (sizeof(gd_float) == sizeof(float)) ? *(float*)v :0); case GDEXTENSION_VARIANT_TYPE_STRING: { auto str = gdu_string_pdup((gd_string*)v); auto j = janet_stringv((void*)str.v, str.sz); _free(str.v); return j; }; case GDEXTENSION_VARIANT_TYPE_STRING_NAME: { /* we can reasonably assume syms will be small enough * to fit on the stack and avoid a pointless malloc */ auto str = _gdu_stringName_stackp((gd_stringName*)v); auto j = janet_keywordv((void*)str.v, str.sz); return j; }; case GDEXTENSION_VARIANT_TYPE_ARRAY: { auto sz = gd_array_size(v); auto ja = janet_array(sz); for (size_t i = 0; i < sz; ++i) { auto val = _t(array).operatorIndexConst(v, i); auto j = gdjn_janetize(val); janet_array_push(ja, j); } return janet_wrap_array(ja); }; default: assert(false); } } typedef struct jn_closure { Janet (*fn)(void* data, int32_t argc, Janet* argv); void (*gc)(void* data); char alignas(max_align_t) data []; } jn_closure; typedef struct jn_hnd_dict { JanetAbstractHead header; GDExtensionVariantType key, val; gd_dictionary dict; } jn_hnd_dict; typedef struct jn_hnd_array { JanetAbstractHead header; GDExtensionVariantType ty; gd_array array; } jn_hnd_array; static int api_del_closure(void* ptr, size_t sz) { jn_closure* c = ptr; if (c -> gc != nullptr) { (*c -> gc)(c -> data); } return 0; } static Janet api_call_closure ( void* ptr, int32_t argc, Janet* argv ) { jn_closure* c = ptr; return (c -> fn)(c -> data, argc, argv); } static int api_del_dict(void* ptr, size_t sz) { jn_hnd_dict* dict = ptr; _t(dictionary).dtor(&dict -> dict); printf("drop dict\n"); return 0; } static int api_del_array(void* ptr, size_t sz) { jn_hnd_array* array = ptr; _t(array).dtor(&array -> array); printf("drop array\n"); return 0; } const JanetAbstractType jn_closure_def = { .name = "closure", .call = api_call_closure, .gc = api_del_closure, }; const JanetAbstractType jn_hnd_dict_def = { .name = "prim/type/dict", .gc = api_del_dict, }; const JanetAbstractType jn_hnd_array_def = { .name = "prim/type/array", .gc = api_del_array, }; static Janet api_new_array(int32_t argc, Janet* argv) { auto a = (jn_hnd_array*)janet_abstract(&jn_hnd_array_def, sizeof(jn_hnd_array)); _t(array).empty(&a -> array, nullptr); printf("create array\n"); return janet_wrap_abstract(a); } static Janet api_new_dict(int32_t argc, Janet* argv) { auto a = (jn_hnd_dict*)janet_abstract(&jn_hnd_dict_def, sizeof(jn_hnd_dict)); _t(dictionary).empty(&a -> dict, nullptr); printf("create dict\n"); return janet_wrap_abstract(a); } /* (prim/class-load [<ident> [<ident2>...]]) * low-level class loader. run at compile time to * import a godot class */ static Janet api_class_load(int32_t argc, Janet* argv) { return janet_wrap_nil(); /* FIXME */ } static const JanetReg reg_core [] = { {"class-load", api_class_load, "(prim/class-load ident)\n\n" "low-level loading function for Godot classes"}, {"type/array", api_new_array, "(prim/type/array [...])\n\n" "create a handle to a new godot array object"}, {"type/dict", api_new_dict, "(prim/type/dict <key-type> <val-type> {...})\n\n" "create a handle to a new godot dictionary object"}, {} }; JanetTable* gdjn_vm_api_spawnEnv (JanetTable* api) { /* create a clean new environment that can be used * and discarded by a script without contaminating * the global environment(s) * yes this is ooky */ auto env = jnu_table_extend(api, 8); auto sym_mc = janet_csymbolv("module/cache"); auto cleancache = jnu_table_extend( janet_unwrap_table(janet_table_get(api, sym_mc)), 8); janet_table_put(env, janet_csymbolv("module/cache"), janet_wrap_table(cleancache)); return env; } gdjn_vm_bind gdjn_vm_meta ( JanetTable* bind ) { gdjn_vm_bind b = {}; if (gdjn_vm_metaFlag(bind, "private")) goto fail; if (!gdjn_vm_metaKey(bind, "value", &b.val)) goto fail; if (gdjn_vm_metaFlag(bind, "macro")) b.kind = gdjn_vm_bind_mac; else { if (gdjn_vm_metaKey(bind, "method", &b.meta)) /* TODO assert callability */ b.kind = gdjn_vm_bind_method; else if (gdjn_vm_metaFlag(bind, "class")) b.kind = gdjn_vm_bind_class; else if (gdjn_vm_metaKey(bind, "type", &b.meta)) { if (gdjn_vm_metaFlag(bind, "prop")) { b.kind = gdjn_vm_bind_prop; } else /* constant */ { b.kind = gdjn_vm_bind_const; goto succeed; } if (gdjn_vm_metaFlag(bind, "share")) b.kind |= gdjn_vm_bind_flag_static; } else { switch (janet_type(b.val)) { case JANET_ABSTRACT: if ((janet_abstract_type(&b.val)) != &jn_closure_def) { b.kind = gdjn_vm_bind_const; break; } case JANET_FUNCTION: case JANET_CFUNCTION: b.kind = gdjn_vm_bind_method_static; break; default: goto fail; } } } /* found a valid export, return it */ succeed: return b; /* this binding is not marked correctly for gdexport */ fail: return (gdjn_vm_bind){gdjn_vm_bind_none}; } gdjn_vm_bind gdjn_vm_resv ( JanetTable* env, Janet key ) { auto gchnd = janet_gclock(); Janet def = janet_table_get(env, key); if (janet_type(def) == JANET_NIL) return (gdjn_vm_bind){}; auto m = gdjn_vm_meta(janet_unwrap_table(def)); janet_gcunlock(gchnd); return m; } void gdjn_vm_api_installCommon (JanetTable* tgt) { /* install primitives */ janet_cfuns(tgt, "prim", reg_core); auto idmap = janet_env_lookup(tgt); int gc = janet_gclock(); /* unpack API image */ Janet apiEnv = janet_unmarshal( gdjn_rsrc_api_jimage, sizeof gdjn_rsrc_api_jimage, 0, idmap, nullptr); printf("apienv type is %s\n", janet_type_names[janet_type(apiEnv)]); JanetTable* apiTbl = janet_unwrap_table(apiEnv); /* call the init function to precache base modules */ Janet initDef = janet_table_get(apiTbl, janet_csymbolv("init")); if (janet_type(initDef) == JANET_NIL) { _err("no init fn in api envtbl"); goto fail; } auto initFn = janet_unwrap_function( janet_table_get(janet_unwrap_table(initDef), janet_ckeywordv("value"))); Janet ret; auto e = janet_pcall(initFn, 0, nullptr, &ret, nullptr); if (e == JANET_SIGNAL_ERROR) { _err("failed to unpack the janet environment"); goto fail; } printf("environment load complete\n"); fail: janet_gcunlock(gc); /* janet_collect(); */ } JanetTable* gdjn_vm_api_build_compTime(void) { auto core = janet_core_env(nullptr); auto api = jnu_table_extend(core,32); gdjn_vm_api_installCommon(api); return api; } JanetTable* gdjn_vm_api_build_core(void) { auto core = janet_core_env(nullptr); auto api = jnu_table_extend(core,32); gdjn_vm_api_installCommon(api); return api; } JanetTable* gdjn_vm_compile ( pstr const body, JanetTable* api, const char* ctx ) { if (!ctx) ctx = "<anon>"; if (!api) api = gdjn_ctx -> jn.api; auto cls = jnu_table_extend(api, 8); janet_gcroot(janet_wrap_table(cls)); /* printf("janet: doing bytes %d %s\n", (int)me -> src.sz, me -> src.v); */ int e = janet_dobytes(cls, (uint8_t const*)body.v, body.sz, ctx, nullptr); /* we discard the return value; the environment is what * we're really interested in here */ /* printf("janet: bytes done, got %d\n",e); */ if (e != 0) { _err("janet module could not be loaded"); /* TODO capture parse error */ janet_gcunroot(janet_wrap_table(cls)); cls = nullptr; } return cls; } pstr gdjn_vm_image ( JanetTable* env, JanetTable* binds ) { return (pstr){}; //TODO } |
Added src/vm.h version [dce8093e3b].
> > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 |
#pragma once #include "gdjn.h" #include "type.h" Janet gdjn_janetize_typed ( GDExtensionTypePtr const v, GDExtensionVariantType const t ); static inline Janet gdjn_janetize(gd_variant* const v) { return gdjn_janetize_typed(v, _t(variant).getType(v)); } void gdjn_dejanetize_typed ( GDExtensionTypePtr v, /* out */ GDExtensionVariantType const t, Janet val /* in */ ); // typedef struct gdjn_vm_class { // // } gdjn_vm_class; JanetTable* gdjn_vm_api_build_compTime(void); JanetTable* gdjn_vm_api_build_core(void); JanetTable* gdjn_vm_compile ( pstr const body, JanetTable* api, const char* ctx ); pstr gdjn_vm_image ( JanetTable* env, JanetTable* binds ); typedef struct gdjn_vm_bind { enum gdjn_vm_bind_kind { gdjn_vm_bind_flag_static = 1 << 5, gdjn_vm_bind_flag_doc = 1 << 6, gdjn_vm_bind_flag_umask = ~( gdjn_vm_bind_flag_static | gdjn_vm_bind_flag_doc ), gdjn_vm_bind_none = 0, gdjn_vm_bind_const, gdjn_vm_bind_prop, gdjn_vm_bind_method, gdjn_vm_bind_class, gdjn_vm_bind_mac, /* always static unfortunately */ gdjn_vm_bind_prop_static = gdjn_vm_bind_prop | gdjn_vm_bind_flag_static, gdjn_vm_bind_method_static = gdjn_vm_bind_method | gdjn_vm_bind_flag_static, } kind; Janet val, meta; } gdjn_vm_bind; static inline bool gdjn_vm_metaKey ( JanetTable* const bind, char const* const kw, Janet * valSlot ) { Janet jv; if (valSlot == nullptr) valSlot = &jv; *valSlot = janet_table_get(bind, janet_ckeywordv(kw)); return (janet_type(*valSlot) != JANET_NIL); } static inline bool gdjn_vm_metaFlag ( JanetTable* const bind, char const* const kw ) { Janet v = janet_table_get(bind, janet_ckeywordv(kw)); return janet_truthy(v); } gdjn_vm_bind gdjn_vm_meta ( JanetTable* bind ); gdjn_vm_bind gdjn_vm_resv ( JanetTable* env, Janet key ); |
Modified tool/api-compile.janet from [bd52030fd5] to [56067a22ef].
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 |
(defn api-parse [src] {} #TODO parse json ) (defn api-gen [api] @{} #TODO gen bindings ) (defn main [_ api-src api-dest & _] (def api (with [fd (file/open api-src :r)] (api-gen (api-parse (:read fd :all))))) (def api-bin (make-image api)) (with [fd (file/open api-dest :w)] (:write fd api-bin)) 0) |
| | > > > > | > > > < < > > > > > > > > > > > | > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > | | > > > | < < < | < |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 |
# [ʞ] tool/api-compile.janet # ~ lexi hale <lexi@hale.su> # 🄯 AGPLv3 # ? gathers the core modules and assembles # them into a unified image that can be # loaded into gdjn. (def- extern-table @{}) (def marshal-map @{}) (def unmarshal-map @{}) (defn- safe-to-export? [x] # only ref types and explicit exportscan be safely replaced; # otherwise e.g. any use of the string "release" will be # replaced by the value of janet/build when loaded # yes this is lunacy i don't know why they did it like that (or (function? x) (cfunction? x) (table? x) (array? x) (buffer? x) (has-key? extern-table x))) (defn- register-env [env] (merge-into extern-table (tabseq [[sym data] :pairs env :when (has-key? data :value) :when (safe-to-export? (data :value))] (data :value) sym))) # provide symbol mapping for core values & # external symbols provided by gdjn (register-env root-env) (def modules '[ core prim json ]) # here we iterate over the core modules and load each # in turn. the array *gd-api-map* is assembled at # compile-time and then used by the init function at # runtime to enumerate and install the core modules. # when core modules use a primitive, they declare their # own *gd-api-map* which maps gensyms to the name of # the desired primitive. (def *gd-api-map* @{}) (defn- install [name env] (put *gd-api-map* name env) (when (has-key? env '*gd-api-map*) (def gdmap (get-in env ['*gd-api-map* :value])) (loop [[val key] :pairs gdmap :when (safe-to-export? val)] (def sym (symbol '-val- (bxor (hash name) (hash val)))) (put marshal-map val sym) (put unmarshal-map sym val)) (merge-into extern-table gdmap))) (defn- log [fmt & r] (:write stderr "(api-compile) " (string/format fmt ;r))) (defn- install-mod [name] (log "loading library %s\n" name) (def env (require (string "/lib/" name))) (install (string name) env)) (each m modules (install-mod m)) # macro implementations can be provided directly in janet # and incorporated into the API image without having to # implement anything beyond the primitive API in C # this function is exported as part of api.jimage. # it will be called from vm.c after initializing # the primitive bindings (defn init [] (print "beginning merge") (merge-into module/cache *gd-api-map*) (print "api loaded")) # this is the build-time entry point, which does # the work of assembling the modules and marshalling # them out. it is not exported (defn main [_ out-path] (def env @{}) (each sym '[init marshal-map unmarshal-map] (put env sym ((curenv) sym))) (let [blob (marshal env extern-table)] (with [fd (file/open out-path :w)] (:write fd blob))) 0) |
Modified tool/c-bind-gen.janet from [eeaa9e3ddf] to [37b502b2c6].
72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 ... 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 ... 198 199 200 201 202 203 204 205 206 207 208 209 210 211 ... 224 225 226 227 228 229 230 231 232 233 234 235 236 237 ... 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 ... 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 ... 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 ... 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 ... 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 |
[ ;((<gd-sym> :%init) spec) :binds (map bind-def (or (spec :binds) [])) :methods (map bind-def (or (spec :methods) [])) :ops (map bind-def (or (spec :ops) [])) :ctors (or (spec :ctors) {}) :mode (spec :mode)]) :binds [] :methods [] :ctors {} :ops [] :mode : :enum (fn [me] (string "GDEXTENSION_VARIANT_TYPE_" (:scream me))) )) (defn env: [v dflt] (or ((os/environ) v) dflt)) ................................................................................ (has-value? x))) (def variants (map |(:@new <gd-type> $) ~[ {:id variant :binds [get-ptr-constructor get-ptr-destructor get-ptr-operator-evaluator get-ptr-internal-getter get-ptr-builtin-method get-type booleanize]} {:id bool } {:id color } ,;(map |{:id $} vector-types) ,;(map |{:id $ :methods '[get set size resize fill clear append append_array insert remove-at has is-empty find rfind count reverse slice duplicate] :ctors {:empty []} } packed-types) {:id array :binds [ref set-typed] :ctors {:empty []}} {:id dictionary :binds [set-typed operator-index operator-index-const] :ctors {:empty []}} {:id string-name :mode :dc :ops [equal] :binds [new-with-utf8-chars new-with-utf8-chars-and-len] :methods [length ends-with begins-with ................................................................................ get-reference-count]} {:id script :binds [get-source-code set-source-code]} {:id file-access :binds [open close store-string get-as-text]} {:id resource-loader :binds [add-resource-format-loader remove-resource-format-loader]} {:id resource-saver :binds [add-resource-format-saver remove-resource-format-saver]} ])) (def global-enums (map |(:from <gd-sym> $) '[ error ])) (def singletons (map |(:from <gd-sym> $) '[ ................................................................................ (def api { :decls @[] :aliases @[] :calls @[] :defer-calls @[] # dependency Hel bypass :types @[] :method-defs @[] }) (def config (env: "gd_config" "double_64")) (def sizes (do (var sz-list nil) (loop [cfg :in (api-spec "builtin_class_sizes") :until (not= nil sz-list)] (when (= config (cfg "build_configuration")) ................................................................................ (case (sizes x) 4 small 8 big (error (string "bad type size " (sizes x) " for " x)))) (case x "int" (bp "int" "int32_t" "int64_t") "float" (bp "float" "float" "double") "bool" "bool")) (defn variant:gd->c [x] (def v (find |(= x (:tall (:@new <gd-sym> {:id ($ :id)}))) variants)) (string "gd_" (:name v))) (defn translate-type [st &opt flags] (defn fl [x] (string/check-set (or flags :) x)) (match (string/split "::" st) # enums can be directly mapped to a C ................................................................................ (def id (string "gd_" (:tall class) "_" (e "name"))) # the underlying type is IMPORTANT! godot enums # appear to use the godot int type, which (at present) # is always 8 bytes long. this means trying to write # a godot "enum" to a plain old C enum will, if you # are very lucky, cause your program to barf all over # the stack and immediately segfault (ln "typedef enum %s : int64_t {" id) # thank the gods for C23. this would have been really # unpleasant otherwise (each n (e "values") (def ident (:from-snek <gd-sym> (n "name"))) (def sym (:@new <gd-sym> ident)) (ln "\t%s_%s = %d," id (:name sym) (n "value")) (ln "\t/* %s */" ................................................................................ (def name (:from-snek <gd-sym> (v "name"))) (ln "\tgd_%s_%s = %d," (:tall e) (:name name) (v "value")) (when (v "description") (ln "\t/* %s */" (v "description")))) (ln "} gd_%s;\n" (:tall e)) ) (loop [v :in variants :let [vsz (or (sizes (:tall v)) (sizes (string (v :id) )))]] (add (api :aliases) # these are all opaque objects "typedef _opaque(%d) gd_%s;" vsz (:name v)) (add (api :decls ) "struct {") (def my-enum (:enum v)) (add-methods v (v :binds)) (add-enums v) # bind builtins # WHY IS THIS *YET ANOTHER* COMPLETELY DIFFERENT API # for the SAME LITERAL THING fuck youuuuu (when (has-key? gdclasses (:tall v)) (loop [m :in (v :methods) :let [method (get-in gdclasses [(:tall v) :methods (:sulk m)])]] ................................................................................ (translate-type (method :return_type)))) (def args (method:args method (:tall v) (if (method :is_const) :c :))) (def impl @[]) (unless (= return-type "void") (array/push impl (string/format "typeof(%s) ret;" return-type))) (array/push impl (string/format "_g_typeDB -> gd_%s.%s(" (:name v) (:name m))) (array/push impl (string "\t" (if (method :is_static) ................................................................................ (print "\ntypedef struct gdjn_typeDB {") (loop [d :in (api :decls)] (print "\t" d)) (print "} gdjn_typeDB;") (each t (api :types) (print t)) (print c-fetch-decl ";") (each m (api :method-defs) (print (m :dfn) ";"))) "loader" (do (print "#include <stdlib.h>\n" "#include <stdio.h>\n" "#include <assert.h>\n" "#include \"interface.h\"\n\n" "static gdjn_typeDB* _g_typeDB;\n\n" # HORRID HACK c-fetch-decl "{\n" ;(map |(string "\t" $ "\n") [ "_g_typeDB = t;" ;(api :calls) ;(api :defer-calls) ]) "}") (each m (api :method-defs) (print (m :dfn) "{\n\t" (string/join (m :impl) "\n\t") "\n}\n"))))) |
> | > > > > > > | > > | | | | > | > | > | > > > > > | > | > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > > | | > > | > > > |
72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 ... 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 ... 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 ... 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 ... 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 ... 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 ... 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 ... 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 ... 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 |
[ ;((<gd-sym> :%init) spec) :binds (map bind-def (or (spec :binds) [])) :methods (map bind-def (or (spec :methods) [])) :ops (map bind-def (or (spec :ops) [])) :indexed (or (spec :indexed) false) :ctors (or (spec :ctors) {}) :mode (spec :mode) :c-repr (spec :c-repr)]) :binds [] :methods [] :ctors {} :ops [] :mode : :c-type (fn [me] (string (or (me :c-repr) (string "gd_" (:name me))))) :enum (fn [me] (string "GDEXTENSION_VARIANT_TYPE_" (:scream me))) )) (defn env: [v dflt] (or ((os/environ) v) dflt)) ................................................................................ (has-value? x))) (def variants (map |(:@new <gd-type> $) ~[ {:id variant :binds [get-ptr-constructor get-ptr-destructor get-ptr-operator-evaluator get-ptr-internal-getter get-ptr-builtin-method get-indexed set-indexed get-named set-named get-type booleanize]} {:id bool } {:id int } {:id float } {:id color } ,;(map |{:id $ :mode :dc} vector-types) ,;(map |{:id $ :mode :dc :methods '[get set size resize fill clear append append_array insert remove-at has is-empty find rfind count reverse slice duplicate] :ctors {:empty []} } packed-types) {:id array :mode :dc :binds [ref set-typed operator-index operator-index-const] :methods [size is-empty clear] :ctors {:empty []} :indexed true} {:id dictionary :mode :dc :binds [set-typed operator-index operator-index-const] :methods [size is-empty clear] :ctors {:empty []} :indexed true} {:id string-name :mode :dc :ops [equal] :binds [new-with-utf8-chars new-with-utf8-chars-and-len] :methods [length ends-with begins-with ................................................................................ get-reference-count]} {:id script :binds [get-source-code set-source-code]} {:id file-access :binds [open close store-string get-as-text]} {:id resource-loader :binds [add-resource-format-loader remove-resource-format-loader]} {:id resource-saver :binds [add-resource-format-saver remove-resource-format-saver]} {:id script-language-extension} {:id script-extension} ])) (def global-enums (map |(:from <gd-sym> $) '[ error ])) (def singletons (map |(:from <gd-sym> $) '[ ................................................................................ (def api { :decls @[] :aliases @[] :calls @[] :defer-calls @[] # dependency Hel bypass :types @[] :method-defs @[] :methods-inline @[] :wrappers @[] }) (def config (env: "gd_config" "double_64")) (def sizes (do (var sz-list nil) (loop [cfg :in (api-spec "builtin_class_sizes") :until (not= nil sz-list)] (when (= config (cfg "build_configuration")) ................................................................................ (case (sizes x) 4 small 8 big (error (string "bad type size " (sizes x) " for " x)))) (case x "int" (bp "int" "int32_t" "int64_t") "float" (bp "float" "float" "double") "bool" "bool" "object" "GDExtensionObjectPtr")) (defn variant:gd->c [x] (def v (find |(= x (:tall (:@new <gd-sym> {:id ($ :id)}))) variants)) (string "gd_" (:name v))) (defn translate-type [st &opt flags] (defn fl [x] (string/check-set (or flags :) x)) (match (string/split "::" st) # enums can be directly mapped to a C ................................................................................ (def id (string "gd_" (:tall class) "_" (e "name"))) # the underlying type is IMPORTANT! godot enums # appear to use the godot int type, which (at present) # is always 8 bytes long. this means trying to write # a godot "enum" to a plain old C enum will, if you # are very lucky, cause your program to barf all over # the stack and immediately segfault (ln "typedef enum %s : GDExtensionInt {" id) # thank the gods for C23. this would have been really # unpleasant otherwise (each n (e "values") (def ident (:from-snek <gd-sym> (n "name"))) (def sym (:@new <gd-sym> ident)) (ln "\t%s_%s = %d," id (:name sym) (n "value")) (ln "\t/* %s */" ................................................................................ (def name (:from-snek <gd-sym> (v "name"))) (ln "\tgd_%s_%s = %d," (:tall e) (:name name) (v "value")) (when (v "description") (ln "\t/* %s */" (v "description")))) (ln "} gd_%s;\n" (:tall e)) ) (add (api :wrappers) (string "auto getWrap = " "(GDExtensionInterfaceGetVariantFromTypeConstructor)" `getProc("get_variant_from_type_constructor");` "\nauto getCast = " "(GDExtensionInterfaceGetVariantToTypeConstructor)" `getProc("get_variant_to_type_constructor");` "\n")) (loop [v :in variants :let [vsz (or (sizes (:tall v)) (sizes (string (v :id) )))]] (add (api :aliases) # these are all opaque objects "typedef _opaque(%d) gd_%s;" vsz (:name v)) (add (api :decls ) "struct {") (def my-enum (:enum v)) (add-methods v (v :binds)) (add-enums v) (unless (= (v :id) 'variant) (add (api :decls) "\tGDExtensionVariantFromTypeConstructorFunc wrap;") (add (api :decls) "\tGDExtensionTypeFromVariantConstructorFunc cast;") (add (api :wrappers) "t -> gd_%s.wrap = getWrap(%s);" (:name v) (:enum v)) (add (api :wrappers) "t -> gd_%s.cast = getCast(%s);" (:name v) (:enum v)) (def ct (or (prim:gd->c (:name v)) (variant:gd->c (:tall v)))) (add (api :methods-inline) (string "static inline %s\n" "gd_variant_to_%s\n" "(\tgd_variant const* const v\n" ") {\n" "\textern struct gdjn_typeDB* _g_typeDB;" "\t%s ret = {};\n" "\t_g_typeDB -> gd_%s.cast(&ret, (void*)v);\n" "\treturn ret;\n" "}\n") ct (:name v) ct (:name v) ) (add (api :methods-inline) (string "static inline gd_variant\n" "gd_variant_of_%s\n" "(\t%s v\n" ") {\n" "\textern struct gdjn_typeDB* _g_typeDB;" "\tgd_variant ret = {};\n" "\t_g_typeDB -> gd_%s.wrap(&ret, &v);\n" "\treturn ret;\n" "}\n") (:name v) ct (:name v) )) # bind builtins # WHY IS THIS *YET ANOTHER* COMPLETELY DIFFERENT API # for the SAME LITERAL THING fuck youuuuu (when (has-key? gdclasses (:tall v)) (loop [m :in (v :methods) :let [method (get-in gdclasses [(:tall v) :methods (:sulk m)])]] ................................................................................ (translate-type (method :return_type)))) (def args (method:args method (:tall v) (if (method :is_const) :c :))) (def impl @[]) (unless (= return-type "void") (array/push impl (string/format "typeof(%s) ret = {};" return-type))) (array/push impl (string/format "_g_typeDB -> gd_%s.%s(" (:name v) (:name m))) (array/push impl (string "\t" (if (method :is_static) ................................................................................ (print "\ntypedef struct gdjn_typeDB {") (loop [d :in (api :decls)] (print "\t" d)) (print "} gdjn_typeDB;") (each t (api :types) (print t)) (print c-fetch-decl ";") (each m (api :method-defs) (print (m :dfn) ";")) (each m (api :methods-inline) (print m))) "loader" (do (print "#include <stdlib.h>\n" "#include <stdio.h>\n" "#include <assert.h>\n" "#include \"interface.h\"\n\n" "gdjn_typeDB* _g_typeDB;\n\n" # HORRID HACK c-fetch-decl "{\n" ;(map |(string "\t" $ "\n") [ "_g_typeDB = t;" ;(api :calls) "{" ;(api :wrappers) "}" ;(api :defer-calls) ]) "}") (each m (api :method-defs) (print (m :dfn) "{\n\t" (string/join (m :impl) "\n\t") "\n}\n"))))) |
Modified tool/class-compile.janet from [a0ecf192d6] to [22543b9ce7].
33 34 35 36 37 38 39 40 41 42 43 44 45 46 ... 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 ... 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 ... 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 ... 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 ... 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 ... 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 ... 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 ... 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 ... 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 ... 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 ... 997 998 999 1000 1001 1002 1003 1004 1005 1006 |
(<- (any (+ (* `\` (backmatch :quo)) (if-not (backmatch :quo) 1)))) (backmatch :quo)) :quo))) (defn fail [cause] (defn mk [ln col] {:kind :parse :cause cause :ln ln :col col}) ~(error (cmt (* (line) (column)) ,mk))) (defn req [id body] ~(+ (* ,;body) ,(fail (keyword :malformed- id)))) (defn kw [id body] ~(* ,id (> 1 :bound) ,(req id ~(:s+ ,;body)))) (defn comment-syntax [id open close] ................................................................................ (enter body "")) (defn err->msg [e] (defn err-span [h & m] [:span [:hl [:red h]] ": " ;m ]) (match e {:kind :parse} (err-span "parse error" (string (e :cause) " at ") [:hl (string/format "%d:%d" (e :ln) (e :col))]) _ (error e))) # _ (err-span "error" # "something went wrong (" [:em (string e)] ")"))) (defn indent [n lst] (map |(string (string/repeat "\t" n) $) lst)) ................................................................................ [:ref _] :object x (keyword x))) (defn gdtype->ctype [t] (match t :void :void :float :double :int :int64_t :bool :bool [:array _] :gd_array [:dictionary _] :gd_dictionary [:ref _] :GDExtensionObjectPtr # use an opaque wrapper struct defined in interface.h x (string "gd_" (:say (:read <sym> t))))) ................................................................................ ~(do (var ,acc ,init) (loop ,binds (set ,acc (do ,;body))) ,acc)) (defn unit-files [u] (def h @[ `#pragma once` `#include "gdjn.h"` `#include "util.h"` ;(u :header-prefix)]) (def c @[ (string `#include "` (:stab (u :name)) `.h"`) `typedef struct gdjn_vcall {` ` GDExtensionClassMethodPtrCall caller; ` ` void* tgt;` `} gdjn_vcall;` ................................................................................ :method (array/push (cls :methods) meth) :impl (array/push (cls :impls) meth))) ))) (def root (:new <cursor> :unit unit)) (each n ast (process n root)) (defn class-prefix [c & r] (def pf (let [p (:prefix c)] (if (empty? p) [] [p]))) (string/join ["gdjn_class" ;pf ;r] "_")) (defn bind-methods [class] (defn bind [f kind] (def t-args (map (fn [[t] &] t) (f :args))) (def invocant ((unit :invocants) [(f :ret) t-args])) (def invocant-ptr ((unit :invocants-ptr) [(f :ret) t-args])) (def fp-t (func-ptr-t (gdtype->ctype (f :ret)) [(string "typeof(" (class-prefix (class :cursor) (class :id))")*") ;(map gdtype->ctype t-args)])) (def strings-lst @[]) (def strings (cache (fn [idx text] (def id (string "_priv_str_" idx)) (array/push strings-lst id text) ................................................................................ `,`]) :impl [`.return_value = ` ;(prop-info (if (= :void (f :ret)) :nil (f :ret)))] )]) ) (def fn-path (class-prefix (f :cursor) "method" (f :id))) (with-names [:s_methodName (f :id) ;strings-lst] (if (not= kind :method) "" (string fp-t " func = " fn-path ";")) (string/format `auto info = (%s) {` (case kind :method "GDExtensionClassMethodInfo" :impl "GDExtensionClassVirtualMethodInfo")) ................................................................................ (length t-args) ",") ;(indent 1 arg-info) ;(if (= kind :method) [ ` .method_userdata = func,` ` .method_flags = GDEXTENSION_METHOD_FLAGS_DEFAULT,` (string "\t.has_return_value = " (if (= :void (f :ret)) "false" "true") ",") (string "\t.call_func = " invocant ",") (string "\t.ptrcall_func = " invocant-ptr ",") ] []) ;ret-info `};` (comment (string `printf("binding method %s\n",` (string/format "%q" fn-path) `);`)) (string `_t(classdb).` (case kind :method "registerExtensionClassMethod" :impl "registerExtensionClassVirtualMethod") `(gdjn_ctx -> gd.lib, &s_className, &info);`))) (array/concat @[] "{" ;(with-names [:s_className (class :id)] ;(map |(bind $ :method) (class :methods)) ;(map |(bind $ :impl) (class :impls))) "}")) (defn push-item [kind cursor item] # abuse hashtables for fun & profit (array/push (unit kind) item) (when (and cursor (cursor :doc)) (put (unit :doc) item (cursor :doc)))) (loop [c :in (unit :classes)] (def id (class-prefix (c :cursor) (c :id))) (def [id-ctor id-dtor id-ctor-api id-init id-create] (map |(string id "_" $) ["new" "del" "api_new" "init" "create"])) (def id-base (as-> (c :base) b ................................................................................ (string "gdjn_class_" b))) #HAAACK (when (not (empty? (c :impls))) (def vtbl @[]) (loop [i :range [0 (length (c :impls))] :let [f ((c :impls) i)]] (def t-args (map (fn [[t] &] t) (f :args))) (def call (class-prefix (f :cursor) "method" (f :id))) (def caller ((unit :invocants-ptr) [(f :ret) t-args])) (put (c :vtbl-map) (f :id) i) (array/push vtbl (string "{.caller=" caller ", .tgt=" call "}")) ) (let [vstr (string/join (map |(string "\n\t" $ ",") vtbl)) vwr (string "{" vstr "\n}") ] ................................................................................ (with-names ["superName" (c :base)] "super = _t(classdb).constructObject(&superName);") [(string/format "super = %s_create();" id-base)]) "return super;")) (def icb-null "(&(GDExtensionInstanceBindingCallbacks){})") (array/push (unit :funcs) (:dec <func-c> self-ref-t id-ctor [] (string "typeof("id")* me = _alloc("id", 1);") ;(with-names ["className" (c :id)] (string/format "auto gdobj = %s();" id-create) #`printf("constructed super object %p\n", gdobj);` "_t(object).setInstance(gdobj, &className, me);" # register the instance as a binding so # that other classes can access it. this # is so dumb ................................................................................ (string id-init "(me, gdobj);")) "return me;")) (push-event :dtor id-dtor :void [[:void* :_data] [:GDExtensionClassInstancePtr :_ptr_me]] |[(string "typeof("id")* me = _ptr_me;") ;$ "_free(me);"]) (def id-virt (class-prefix (c :cursor) (c :id) "virt")) (array/push (unit :funcs) (:dec* <func-c> [:inline :static] self-ref-t (string id "_data") [[:GDExtensionObjectPtr :self]] "return _t(object).getInstanceBinding(" " self, gdjn_ctx -> gd.lib," icb-null ");")) (array/push (unit :funcs) (:dec- <func-c> :void* id-virt [[:void* :data] [:GDExtensionConstStringNamePtr :method] [:uint32_t :hash]] `bool res = false;` ;(catseq [[name idx] :pairs (c :vtbl-map)] [ ;(with-names [:name name] `_t(stringName).equal(&name, method, &res);` `if (res) {` (string "\treturn (void*)&" id "_vtbl[" idx "];") `}`)]) ;(if (= :native (c :base-mode)) [`return nullptr;`] # inherits from a gdextension class; call up [(string/format "return %s_virt(data, method, hash);" id-base)]) )) (def id-virt-call (class-prefix (c :cursor) (c :id) "virt_call")) (array/push (unit :funcs) (:dec- <func-c> :void id-virt-call [[:GDExtensionClassInstancePtr :inst] [:GDExtensionConstStringNamePtr :method] [:void* :vcall] ["const GDExtensionConstTypePtr*" :args] [:GDExtensionTypePtr :ret]] `auto c = (const gdjn_vcall*)vcall;` `c -> caller(c -> tgt, inst, args, ret);`)) (array/push (unit :load) ;(with-names ["className" (c :id) "superName" (c :base)] "auto classDesc = (GDExtensionClassCreationInfo4) {" ` .is_virtual = false,` (string "\t.is_abstract = " (if (c :abstract) "true" "false")",") ` .is_exposed = true,` ` .is_runtime = true,` (string "\t.create_instance_func = " id-ctor-api ",") ................................................................................ (def binder (bind-methods c)) (array/concat (unit :load) binder) ) (loop [f :in (unit :methods)] (def class (class-prefix (f :cursor))) (def cid (class-prefix (f :cursor) "method" (f :id))) (def cfn (:dec <func-c> (gdtype->ctype (f :ret)) cid [ [(string class "*") "me"] ;(map (fn [[t id dox]] [(gdtype->ctype t) id]) (f :args))] (f :text))) (def arg-dox ................................................................................ (let [uf (unit-files unit)] (:write stdout (case emit "header" (lines->str (uf :header)) "loader" (lines->str (uf :impl)) (error :bad-cmd))))) (defn main [& argv] # (entry ;argv)) (try (entry ;argv) ([e] (:write stderr (style ;(err->msg e)))))) |
> | | | | | > > > > | | | | | | | > | | | > > | | > > | < | > | | | | | | > | | | | | | |
33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 ... 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 ... 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 ... 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 ... 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 ... 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 ... 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 ... 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 ... 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 ... 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 ... 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 .... 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 |
(<- (any (+ (* `\` (backmatch :quo)) (if-not (backmatch :quo) 1)))) (backmatch :quo)) :quo))) (defn fail [cause] (defn mk [ln col] {:kind :parse :cause cause :src (dyn *src-file* "<stdin>") :ln ln :col col}) ~(error (cmt (* (line) (column)) ,mk))) (defn req [id body] ~(+ (* ,;body) ,(fail (keyword :malformed- id)))) (defn kw [id body] ~(* ,id (> 1 :bound) ,(req id ~(:s+ ,;body)))) (defn comment-syntax [id open close] ................................................................................ (enter body "")) (defn err->msg [e] (defn err-span [h & m] [:span [:hl [:red h]] ": " ;m ]) (match e {:kind :parse} (err-span "parse error" (string (e :cause) " at ") [:hl (string/format "%s:%d:%d" (e :src) (e :ln) (e :col))]) _ (error e))) # _ (err-span "error" # "something went wrong (" [:em (string e)] ")"))) (defn indent [n lst] (map |(string (string/repeat "\t" n) $) lst)) ................................................................................ [:ref _] :object x (keyword x))) (defn gdtype->ctype [t] (match t :void :void :float :double :int :int64_t :bool :uint8_t [:array _] :gd_array [:dictionary _] :gd_dictionary [:ref _] :GDExtensionObjectPtr # use an opaque wrapper struct defined in interface.h x (string "gd_" (:say (:read <sym> t))))) ................................................................................ ~(do (var ,acc ,init) (loop ,binds (set ,acc (do ,;body))) ,acc)) (defn unit-files [u] (def h @[ `#pragma once` `#include "gdjn.h"` ;(u :header-prefix)]) (def c @[ (string `#include "` (:stab (u :name)) `.h"`) `typedef struct gdjn_vcall {` ` GDExtensionClassMethodPtrCall caller; ` ` void* tgt;` `} gdjn_vcall;` ................................................................................ :method (array/push (cls :methods) meth) :impl (array/push (cls :impls) meth))) ))) (def root (:new <cursor> :unit unit)) (each n ast (process n root)) (defn class-prefix* [sep c begin r] (def pf (let [p (:prefix c)] (if (empty? p) [] [p]))) (string/join [;begin ;pf ;r] sep)) (defn class-prefix. [c & r] (class-prefix* "." c [] r)) (defn class-prefix_ [c & r] (class-prefix* "_" c ["gdjn_class"] r)) (defn bind-methods [class] (defn bind [f kind] (def t-args (map (fn [[t] &] t) (f :args))) (def invocant |((unit :invocants) [(f :ret) t-args])) (def invocant-ptr |((unit :invocants-ptr) [(f :ret) t-args])) (def fp-t (func-ptr-t (gdtype->ctype (f :ret)) [(string "typeof(" (class-prefix_ (class :cursor) (class :id))")*") ;(map gdtype->ctype t-args)])) (def strings-lst @[]) (def strings (cache (fn [idx text] (def id (string "_priv_str_" idx)) (array/push strings-lst id text) ................................................................................ `,`]) :impl [`.return_value = ` ;(prop-info (if (= :void (f :ret)) :nil (f :ret)))] )]) ) (def fn-path (class-prefix_ (f :cursor) "method" (f :id))) (with-names [:s_methodName (f :id) ;strings-lst] (if (not= kind :method) "" (string fp-t " func = " fn-path ";")) (string/format `auto info = (%s) {` (case kind :method "GDExtensionClassMethodInfo" :impl "GDExtensionClassVirtualMethodInfo")) ................................................................................ (length t-args) ",") ;(indent 1 arg-info) ;(if (= kind :method) [ ` .method_userdata = func,` ` .method_flags = GDEXTENSION_METHOD_FLAGS_DEFAULT,` (string "\t.has_return_value = " (if (= :void (f :ret)) "false" "true") ",") (string "\t.call_func = " (invocant) ",") (string "\t.ptrcall_func = " (invocant-ptr) ",") ] []) ;ret-info `};` (comment (string `printf("binding method %s\n",` (string/format "%q" fn-path) `);`)) (string `_t(classdb).` (case kind :method "registerExtensionClassMethod" :impl "registerExtensionClassVirtualMethod") `(gdjn_ctx -> gd.lib, &s_className, &info);`))) (def class-path (class-prefix* "_" (class :cursor) [] [(class :id)])) (array/concat @[] "{" ;(with-names [:s_className class-path] ;(map |(bind $ :method) (class :methods)) ;(map |(bind $ :impl) (class :impls))) "}")) (defn push-item [kind cursor item] # abuse hashtables for fun & profit (array/push (unit kind) item) (when (and cursor (cursor :doc)) (put (unit :doc) item (cursor :doc)))) (loop [c :in (unit :classes)] (def id (class-prefix_ (c :cursor) (c :id))) (def [id-ctor id-dtor id-ctor-api id-init id-create] (map |(string id "_" $) ["new" "del" "api_new" "init" "create"])) (def id-base (as-> (c :base) b ................................................................................ (string "gdjn_class_" b))) #HAAACK (when (not (empty? (c :impls))) (def vtbl @[]) (loop [i :range [0 (length (c :impls))] :let [f ((c :impls) i)]] (def t-args (map (fn [[t] &] t) (f :args))) (def call (class-prefix_ (f :cursor) "method" (f :id))) (def caller ((unit :invocants-ptr) [(f :ret) t-args])) (put (c :vtbl-map) (f :id) i) (array/push vtbl (string "{.caller=" caller ", .tgt=" call "}")) ) (let [vstr (string/join (map |(string "\n\t" $ ",") vtbl)) vwr (string "{" vstr "\n}") ] ................................................................................ (with-names ["superName" (c :base)] "super = _t(classdb).constructObject(&superName);") [(string/format "super = %s_create();" id-base)]) "return super;")) (def icb-null "(&(GDExtensionInstanceBindingCallbacks){})") (def class-path (class-prefix* "_" (c :cursor) [] [(c :id)])) (array/push (unit :funcs) (:dec <func-c> self-ref-t id-ctor [] (string `printf("creating object ` id `\n");`) (string "typeof("id")* me = _alloc("id", 1);") ;(with-names ["className" class-path] (string/format "auto gdobj = %s();" id-create) #`printf("constructed super object %p\n", gdobj);` "_t(object).setInstance(gdobj, &className, me);" # register the instance as a binding so # that other classes can access it. this # is so dumb ................................................................................ (string id-init "(me, gdobj);")) "return me;")) (push-event :dtor id-dtor :void [[:void* :_data] [:GDExtensionClassInstancePtr :_ptr_me]] |[(string "typeof("id")* me = _ptr_me;") ;$ (if (= :native (c :base-mode)) "_free(me);" (string/format "%s_del(_data, &me -> super);" id-base))]) (def id-virt (class-prefix_ (c :cursor) (c :id) "virt")) (array/push (unit :funcs) (:dec* <func-c> [:inline :static] self-ref-t (string id "_data") [[:GDExtensionObjectPtr :self]] "return _t(object).getInstanceBinding(" " self, gdjn_ctx -> gd.lib," icb-null ");")) (array/push (unit :funcs) (:dec- <func-c> :void* id-virt [[:void* :data] [:GDExtensionConstStringNamePtr :method] [:uint32_t :hash]] ;(catseq [[name idx] :pairs (c :vtbl-map)] [ `{bool res = false;` ;(with-names [:name name] `_t(stringName).equal(&name, method, &res);`) `if (res) {` (string "\treturn (void*)&" id "_vtbl[" idx "];") `}}`]) ;(if (= :native (c :base-mode)) [`return nullptr;`] # inherits from a gdextension class; call up [(string/format "return %s_virt(data, method, hash);" id-base)]) )) (def id-virt-call (class-prefix_ (c :cursor) (c :id) "virt_call")) (array/push (unit :funcs) (:dec- <func-c> :void id-virt-call [[:GDExtensionClassInstancePtr :inst] [:GDExtensionConstStringNamePtr :method] [:void* :vcall] ["const GDExtensionConstTypePtr*" :args] [:GDExtensionTypePtr :ret]] `auto c = (const gdjn_vcall*)vcall;` `c -> caller(c -> tgt, inst, args, ret);`)) (def class-path (class-prefix* "_" (c :cursor) [] [(c :id)])) (array/push (unit :load) ;(with-names ["className" class-path "superName" (c :base)] "auto classDesc = (GDExtensionClassCreationInfo4) {" ` .is_virtual = false,` (string "\t.is_abstract = " (if (c :abstract) "true" "false")",") ` .is_exposed = true,` ` .is_runtime = true,` (string "\t.create_instance_func = " id-ctor-api ",") ................................................................................ (def binder (bind-methods c)) (array/concat (unit :load) binder) ) (loop [f :in (unit :methods)] (def class (class-prefix_ (f :cursor))) (def cid (class-prefix_ (f :cursor) "method" (f :id))) (def cfn (:dec <func-c> (gdtype->ctype (f :ret)) cid [ [(string class "*") "me"] ;(map (fn [[t id dox]] [(gdtype->ctype t) id]) (f :args))] (f :text))) (def arg-dox ................................................................................ (let [uf (unit-files unit)] (:write stdout (case emit "header" (lines->str (uf :header)) "loader" (lines->str (uf :impl)) (error :bad-cmd))))) (defn main [& argv] (entry ;argv)) # (try (entry ;argv) # ([e] (:write stderr (style ;(err->msg e)) "\n")))) |
Modified tool/rsrc.janet from [5689af4629] to [c425f8a03e].
28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 |
(let [fd (file/open path :r) rec {:id (reduce |(string/replace-all $1 "_" $0) (basename path) ["-" "." "/"]) :vals (blob->c-array (:read fd :all))}] (:close fd) rec)) (defn c-decl [t] (string "const uint8_t gdjn_rsrc_" (t :id) " []")) (defn c-def [t] (string (c-decl t) " = {" (t :vals) "}")) (defn c-compile [c to-path] (def cc (os/spawn [(dyn *cc*) "-xc" "-c" "-" "-o" to-path] :p {:in :pipe})) |
| | > > | |
28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 |
(let [fd (file/open path :r) rec {:id (reduce |(string/replace-all $1 "_" $0) (basename path) ["-" "." "/"]) :vals (blob->c-array (:read fd :all))}] (:close fd) rec)) (defn c-base [t] (string "const uint8_t gdjn_rsrc_" (t :id) " [" (length (t :vals))"]")) (defn c-decl [t] (string "extern " (c-base t))) (defn c-def [t] (string (c-base t) " = {" (t :vals) "}")) (defn c-compile [c to-path] (def cc (os/spawn [(dyn *cc*) "-xc" "-c" "-" "-o" to-path] :p {:in :pipe})) |