mirror of
				https://github.com/janet-lang/janet
				synced 2025-10-25 20:57:40 +00:00 
			
		
		
		
	Merge branch 'redefs-work'
- Change the global binding name from :redefs to :redef - Simplify internal representation of "redefinable bindings" - Store "redefinable bindings in :ref rather than :value inside the environment entries. This makes such bindings more like vars that can't be set rather than defs.
This commit is contained in:
		| @@ -1952,8 +1952,8 @@ | |||||||
|     (def h (in t 0)) |     (def h (in t 0)) | ||||||
|     (def s (in specs h)) |     (def s (in specs h)) | ||||||
|     (def entry (or (dyn h) {})) |     (def entry (or (dyn h) {})) | ||||||
|     (def m (if (entry :redef) (in (entry :value) 0) (entry :value))) |     (def m (do (def r (get entry :ref)) (if r (in r 0) (get entry :value)))) | ||||||
|     (def m? (entry :macro)) |     (def m? (in entry :macro)) | ||||||
|     (cond |     (cond | ||||||
|       s (s t) |       s (s t) | ||||||
|       m? (do (setdyn :macro-form t) (m ;(tuple/slice t 1))) |       m? (do (setdyn :macro-form t) (m ;(tuple/slice t 1))) | ||||||
| @@ -2186,19 +2186,20 @@ | |||||||
|   the file, prints nothing." |   the file, prints nothing." | ||||||
|   [where line col] |   [where line col] | ||||||
|   (if-not line (break)) |   (if-not line (break)) | ||||||
|  |   (unless (string? where) (break)) | ||||||
|   (when-with [f (file/open where :r)] |   (when-with [f (file/open where :r)] | ||||||
|     (def source-code (file/read f :all)) |     (def source-code (file/read f :all)) | ||||||
|     (var index 0) |     (var index 0) | ||||||
|     (repeat (dec line) |     (repeat (dec line) | ||||||
|        (if-not index (break)) |        (if-not index (break)) | ||||||
|        (set index (inc (string/find "\n" source-code index)))) |        (set index (string/find "\n" source-code index)) | ||||||
|  |        (if index (++ index))) | ||||||
|     (when index |     (when index | ||||||
|       (def line-end (string/find "\n" source-code index)) |       (def line-end (string/find "\n" source-code index)) | ||||||
|       (eprint "  " (string/slice source-code index line-end)) |       (eprint "  " (string/slice source-code index line-end)) | ||||||
|       (when col |       (when col | ||||||
|         (+= index col) |         (+= index col) | ||||||
|         (eprint (string/repeat " " (inc col)) "^")) |         (eprint (string/repeat " " (inc col)) "^"))))) | ||||||
|       (eflush)))) |  | ||||||
|  |  | ||||||
| (defn warn-compile | (defn warn-compile | ||||||
|   "Default handler for a compile warning" |   "Default handler for a compile warning" | ||||||
| @@ -3077,10 +3078,10 @@ | |||||||
|   (def bind-type |   (def bind-type | ||||||
|     (string "    " |     (string "    " | ||||||
|             (cond |             (cond | ||||||
|  |               (x :redef) (type (in (x :ref) 0)) | ||||||
|               (x :ref) (string :var " (" (type (in (x :ref) 0)) ")") |               (x :ref) (string :var " (" (type (in (x :ref) 0)) ")") | ||||||
|               (x :macro) :macro |               (x :macro) :macro | ||||||
|               (x :module) (string :module " (" (x :kind) ")") |               (x :module) (string :module " (" (x :kind) ")") | ||||||
|               (x :redef) (type (in (x :value) 0)) |  | ||||||
|               (type (x :value))) |               (type (x :value))) | ||||||
|             "\n")) |             "\n")) | ||||||
|   (def sm (x :source-map)) |   (def sm (x :source-map)) | ||||||
| @@ -3144,7 +3145,7 @@ | |||||||
|   (loop [module-set :in [[root-env] module/cache] |   (loop [module-set :in [[root-env] module/cache] | ||||||
|          module :in module-set |          module :in module-set | ||||||
|          value :in module] |          value :in module] | ||||||
|     (let [check (or (value :ref) (if (value :redef) (in (value :value) 0) (value :value)))] |     (let [check (or (get value :ref) (get value :value))] | ||||||
|       (when (= check x) |       (when (= check x) | ||||||
|         (print-module-entry value) |         (print-module-entry value) | ||||||
|         (set found true) |         (set found true) | ||||||
| @@ -3538,7 +3539,7 @@ | |||||||
| (defn- run-main | (defn- run-main | ||||||
|   [env subargs arg] |   [env subargs arg] | ||||||
|   (if-let [entry (in env 'main) |   (if-let [entry (in env 'main) | ||||||
|            main (if (entry :redef) (in (entry :value) 0) (entry :value))] |            main (or (get entry :value) (in (get entry :ref) 0))] | ||||||
|     (let [thunk (compile [main ;subargs] env arg)] |     (let [thunk (compile [main ;subargs] env arg)] | ||||||
|       (if (function? thunk) (thunk) (error (thunk :error)))))) |       (if (function? thunk) (thunk) (error (thunk :error)))))) | ||||||
|  |  | ||||||
| @@ -3583,7 +3584,6 @@ | |||||||
|                -e code : Execute a string of janet |                -e code : Execute a string of janet | ||||||
|                -E code arguments... : Evaluate  an expression as a short-fn with arguments |                -E code arguments... : Evaluate  an expression as a short-fn with arguments | ||||||
|                -d : Set the debug flag in the REPL |                -d : Set the debug flag in the REPL | ||||||
|                -D : Use redefinable def bindings |  | ||||||
|                -r : Enter the REPL after running all scripts |                -r : Enter the REPL after running all scripts | ||||||
|                -R : Disables loading profile.janet when JANET_PROFILE is present |                -R : Disables loading profile.janet when JANET_PROFILE is present | ||||||
|                -p : Keep on executing if there is a top-level error (persistent) |                -p : Keep on executing if there is a top-level error (persistent) | ||||||
| @@ -3634,7 +3634,6 @@ | |||||||
|              (error (get thunk :error))) |              (error (get thunk :error))) | ||||||
|            math/inf) |            math/inf) | ||||||
|      "d" (fn [&] (set debug-flag true) 1) |      "d" (fn [&] (set debug-flag true) 1) | ||||||
|      "D" (fn [&] (setdyn :redefs true) 1) |  | ||||||
|      "w" (fn [i &] (set warn-level (get-lint-level i)) 2) |      "w" (fn [i &] (set warn-level (get-lint-level i)) 2) | ||||||
|      "x" (fn [i &] (set error-level (get-lint-level i)) 2) |      "x" (fn [i &] (set error-level (get-lint-level i)) 2) | ||||||
|      "R" (fn [&] (setdyn :profilepath nil) 1)}) |      "R" (fn [&] (setdyn :profilepath nil) 1)}) | ||||||
| @@ -3659,14 +3658,18 @@ | |||||||
|             (put env :args subargs) |             (put env :args subargs) | ||||||
|             (put env :lint-error error-level) |             (put env :lint-error error-level) | ||||||
|             (put env :lint-warn warn-level) |             (put env :lint-warn warn-level) | ||||||
|             (if debug-flag (put env :debug true)) |             (when debug-flag | ||||||
|  |               (put env :debug true) | ||||||
|  |               (put env :redef true)) | ||||||
|             (run-main env subargs arg)) |             (run-main env subargs arg)) | ||||||
|           (do |           (do | ||||||
|             (def env (make-env)) |             (def env (make-env)) | ||||||
|             (put env :args subargs) |             (put env :args subargs) | ||||||
|             (put env :lint-error error-level) |             (put env :lint-error error-level) | ||||||
|             (put env :lint-warn warn-level) |             (put env :lint-warn warn-level) | ||||||
|             (if debug-flag (put env :debug true)) |             (when debug-flag | ||||||
|  |               (put env :debug true) | ||||||
|  |               (put env :redef true)) | ||||||
|             (if compile-only |             (if compile-only | ||||||
|               (flycheck arg :exit exit-on-error :env env) |               (flycheck arg :exit exit-on-error :env env) | ||||||
|               (do |               (do | ||||||
| @@ -3692,7 +3695,9 @@ | |||||||
|         (when-let [profile.janet (dyn :profilepath)] |         (when-let [profile.janet (dyn :profilepath)] | ||||||
|             (def new-env (dofile profile.janet :exit true)) |             (def new-env (dofile profile.janet :exit true)) | ||||||
|             (merge-module env new-env "" false)) |             (merge-module env new-env "" false)) | ||||||
|         (if debug-flag (put env :debug true)) |         (when debug-flag | ||||||
|  |           (put env :debug true) | ||||||
|  |           (put env :redef true)) | ||||||
|         (def getter (if raw-stdin getstdin getline)) |         (def getter (if raw-stdin getstdin getline)) | ||||||
|         (defn getchunk [buf p] |         (defn getchunk [buf p] | ||||||
|           (getter (getprompt p) buf env)) |           (getter (getprompt p) buf env)) | ||||||
|   | |||||||
| @@ -238,10 +238,12 @@ JanetSlot janetc_resolve( | |||||||
|             case JANET_BINDING_DEF: |             case JANET_BINDING_DEF: | ||||||
|             case JANET_BINDING_MACRO: /* Macro should function like defs when not in calling pos */ |             case JANET_BINDING_MACRO: /* Macro should function like defs when not in calling pos */ | ||||||
|                 ret = janetc_cslot(binding.value); |                 ret = janetc_cslot(binding.value); | ||||||
|                 if (binding.dynamic) { |                 break; | ||||||
|                     ret.flags |= JANET_SLOT_REF | JANET_SLOT_NAMED | JANET_SLOTTYPE_ANY; |             case JANET_BINDING_DYNAMIC_DEF: | ||||||
|                     ret.flags &= ~JANET_SLOT_CONSTANT; |             case JANET_BINDING_DYNAMIC_MACRO: | ||||||
|                 } |                 ret = janetc_cslot(binding.value); | ||||||
|  |                 ret.flags |= JANET_SLOT_REF | JANET_SLOT_NAMED | JANET_SLOTTYPE_ANY; | ||||||
|  |                 ret.flags &= ~JANET_SLOT_CONSTANT; | ||||||
|                 break; |                 break; | ||||||
|             case JANET_BINDING_VAR: { |             case JANET_BINDING_VAR: { | ||||||
|                 ret = janetc_cslot(binding.value); |                 ret = janetc_cslot(binding.value); | ||||||
|   | |||||||
| @@ -333,24 +333,26 @@ static int defleaf( | |||||||
|                         janet_wrap_tuple(janetc_make_sourcemap(c))); |                         janet_wrap_tuple(janetc_make_sourcemap(c))); | ||||||
|  |  | ||||||
|         int is_redef = 0; |         int is_redef = 0; | ||||||
|         Janet meta_redef = janet_table_get(entry, janet_ckeywordv("redef")); |         Janet redef_kw = janet_ckeywordv("redef"); | ||||||
|  |         Janet meta_redef = janet_table_get(entry, redef_kw); | ||||||
|         if (janet_truthy(meta_redef)) { |         if (janet_truthy(meta_redef)) { | ||||||
|             is_redef = 1; |             is_redef = 1; | ||||||
|         } else if (janet_checktype(meta_redef, JANET_NIL) && janet_truthy(janet_dyn("redefs"))) { |         } else if (janet_checktype(meta_redef, JANET_NIL) && | ||||||
|             janet_table_put(entry, janet_ckeywordv("redef"), janet_wrap_true()); |                    janet_truthy(janet_table_get(c->env, redef_kw))) { | ||||||
|  |             janet_table_put(entry, redef_kw, janet_wrap_true()); | ||||||
|             is_redef = 1; |             is_redef = 1; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (is_redef) { |         if (is_redef) { | ||||||
|             JanetBinding binding = janet_resolve_ext(c->env, sym); |             JanetBinding binding = janet_resolve_ext(c->env, sym); | ||||||
|             JanetArray *ref; |             JanetArray *ref; | ||||||
|             if (janet_checktype(binding.value, JANET_ARRAY)) { |             if (binding.type == JANET_BINDING_DYNAMIC_DEF || binding.type == JANET_BINDING_DYNAMIC_MACRO) { | ||||||
|                 ref = janet_unwrap_array(binding.value); |                 ref = janet_unwrap_array(binding.value); | ||||||
|             } else { |             } else { | ||||||
|                 ref = janet_array(1); |                 ref = janet_array(1); | ||||||
|                 janet_array_push(ref, janet_wrap_nil()); |                 janet_array_push(ref, janet_wrap_nil()); | ||||||
|             } |             } | ||||||
|             janet_table_put(entry, janet_ckeywordv("value"), janet_wrap_array(ref)); |             janet_table_put(entry, janet_ckeywordv("ref"), janet_wrap_array(ref)); | ||||||
|             JanetSlot refslot = janetc_cslot(janet_wrap_array(ref)); |             JanetSlot refslot = janetc_cslot(janet_wrap_array(ref)); | ||||||
|             janetc_emit_ssu(c, JOP_PUT_INDEX, refslot, s, 0, 0); |             janetc_emit_ssu(c, JOP_PUT_INDEX, refslot, s, 0, 0); | ||||||
|         } else { |         } else { | ||||||
|   | |||||||
| @@ -598,14 +598,12 @@ void janet_core_cfuns_ext(JanetTable *env, const char *regprefix, const JanetReg | |||||||
| #endif | #endif | ||||||
|  |  | ||||||
| JanetBinding janet_resolve_ext(JanetTable *env, const uint8_t *sym) { | JanetBinding janet_resolve_ext(JanetTable *env, const uint8_t *sym) { | ||||||
|     Janet ref; |  | ||||||
|     JanetTable *entry_table; |     JanetTable *entry_table; | ||||||
|     Janet entry = janet_table_get(env, janet_wrap_symbol(sym)); |     Janet entry = janet_table_get(env, janet_wrap_symbol(sym)); | ||||||
|     JanetBinding binding = { |     JanetBinding binding = { | ||||||
|         JANET_BINDING_NONE, |         JANET_BINDING_NONE, | ||||||
|         janet_wrap_nil(), |         janet_wrap_nil(), | ||||||
|         JANET_BINDING_DEP_NONE, |         JANET_BINDING_DEP_NONE | ||||||
|         JANET_BINDING_STATIC |  | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     /* Check environment for entry */ |     /* Check environment for entry */ | ||||||
| @@ -628,36 +626,32 @@ JanetBinding janet_resolve_ext(JanetTable *env, const uint8_t *sym) { | |||||||
|         binding.deprecation = JANET_BINDING_DEP_NORMAL; |         binding.deprecation = JANET_BINDING_DEP_NORMAL; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     binding.dynamic = janet_truthy(janet_table_get(entry_table, janet_ckeywordv("redef"))); |     int redef = janet_truthy(janet_table_get(entry_table, janet_ckeywordv("redef"))); | ||||||
|  |     int macro = janet_truthy(janet_table_get(entry_table, janet_ckeywordv("macro"))); | ||||||
|  |     Janet value = janet_table_get(entry_table, janet_ckeywordv("value")); | ||||||
|  |     Janet ref = janet_table_get(entry_table, janet_ckeywordv("ref")); | ||||||
|  |  | ||||||
|     if (!janet_checktype( |     if (macro) { | ||||||
|                 janet_table_get(entry_table, janet_ckeywordv("macro")), |         binding.value = redef ? ref : value; | ||||||
|                 JANET_NIL)) { |         binding.type = redef ? JANET_BINDING_DYNAMIC_MACRO : JANET_BINDING_MACRO; | ||||||
|         binding.value = janet_table_get(entry_table, janet_ckeywordv("value")); |  | ||||||
|         binding.type = JANET_BINDING_MACRO; |  | ||||||
|         return binding; |         return binding; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     ref = janet_table_get(entry_table, janet_ckeywordv("ref")); |     if (!redef && janet_checktype(ref, JANET_ARRAY)) { | ||||||
|     if (janet_checktype(ref, JANET_ARRAY)) { |  | ||||||
|         binding.value = ref; |         binding.value = ref; | ||||||
|         binding.type = JANET_BINDING_VAR; |         binding.type = JANET_BINDING_VAR; | ||||||
|         return binding; |         return binding; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     binding.value = janet_table_get(entry_table, janet_ckeywordv("value")); |     binding.value = redef ? ref : value; | ||||||
|     binding.type = JANET_BINDING_DEF; |     binding.type = redef ? JANET_BINDING_DYNAMIC_DEF : JANET_BINDING_DEF; | ||||||
|     return binding; |     return binding; | ||||||
| } | } | ||||||
|  |  | ||||||
| JanetBindingType janet_resolve(JanetTable *env, const uint8_t *sym, Janet *out) { | JanetBindingType janet_resolve(JanetTable *env, const uint8_t *sym, Janet *out) { | ||||||
|     JanetBinding binding = janet_resolve_ext(env, sym); |     JanetBinding binding = janet_resolve_ext(env, sym); | ||||||
|     if (binding.dynamic) { |     if (binding.type == JANET_BINDING_DYNAMIC_DEF || binding.type == JANET_BINDING_DYNAMIC_MACRO) { | ||||||
|         if (janet_checktype(binding.value, JANET_ARRAY)) { |         *out = janet_array_peek(janet_unwrap_array(binding.value)); | ||||||
|             *out = janet_array_peek(janet_unwrap_array(binding.value)); |  | ||||||
|         } else { |  | ||||||
|             *out = janet_wrap_nil(); |  | ||||||
|         } |  | ||||||
|     } else { |     } else { | ||||||
|         *out = binding.value; |         *out = binding.value; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1779,7 +1779,9 @@ typedef enum { | |||||||
|     JANET_BINDING_NONE, |     JANET_BINDING_NONE, | ||||||
|     JANET_BINDING_DEF, |     JANET_BINDING_DEF, | ||||||
|     JANET_BINDING_VAR, |     JANET_BINDING_VAR, | ||||||
|     JANET_BINDING_MACRO |     JANET_BINDING_MACRO, | ||||||
|  |     JANET_BINDING_DYNAMIC_DEF, | ||||||
|  |     JANET_BINDING_DYNAMIC_MACRO | ||||||
| } JanetBindingType; | } JanetBindingType; | ||||||
|  |  | ||||||
| typedef struct { | typedef struct { | ||||||
| @@ -1791,9 +1793,6 @@ typedef struct { | |||||||
|         JANET_BINDING_DEP_NORMAL, |         JANET_BINDING_DEP_NORMAL, | ||||||
|         JANET_BINDING_DEP_STRICT, |         JANET_BINDING_DEP_STRICT, | ||||||
|     } deprecation; |     } deprecation; | ||||||
|     enum { |  | ||||||
|         JANET_BINDING_STATIC, |  | ||||||
|         JANET_BINDING_DYNAMIC } dynamic; |  | ||||||
| } JanetBinding; | } JanetBinding; | ||||||
|  |  | ||||||
| JANET_API void janet_def(JanetTable *env, const char *name, Janet val, const char *documentation); | JANET_API void janet_def(JanetTable *env, const char *name, Janet val, const char *documentation); | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ | |||||||
|   (def str (string e)) |   (def str (string e)) | ||||||
|   (if x |   (if x | ||||||
|     (when is-verbose (eprintf "\e[32m✔\e[0m %s: %v" (describe e) x)) |     (when is-verbose (eprintf "\e[32m✔\e[0m %s: %v" (describe e) x)) | ||||||
|     (eprintf "\n\e[31m✘\e[0m %s: %v" (describe e) x)) |     (eprintf "\e[31m✘\e[0m %s: %v" (describe e) x)) | ||||||
|   x) |   x) | ||||||
|  |  | ||||||
| (defmacro assert-error | (defmacro assert-error | ||||||
|   | |||||||
| @@ -307,7 +307,7 @@ | |||||||
| (assert (= 1 (dynamicdef1-inc)) "before redefinition with :redef") | (assert (= 1 (dynamicdef1-inc)) "before redefinition with :redef") | ||||||
| (def dynamicdef1 :redef 1) | (def dynamicdef1 :redef 1) | ||||||
| (assert (= 2 (dynamicdef1-inc)) "after redefinition with :redef") | (assert (= 2 (dynamicdef1-inc)) "after redefinition with :redef") | ||||||
| (setdyn :redefs true) | (setdyn :redef true) | ||||||
| (def staticdef2 {:redef false} 0) | (def staticdef2 {:redef false} 0) | ||||||
| (defn staticdef2-inc [] (+ 1 staticdef2)) | (defn staticdef2-inc [] (+ 1 staticdef2)) | ||||||
| (assert (= 1 (staticdef2-inc)) "before redefinition with :redef false") | (assert (= 1 (staticdef2-inc)) "before redefinition with :redef false") | ||||||
| @@ -318,7 +318,7 @@ | |||||||
| (assert (= 1 (dynamicdef2-inc)) "before redefinition with dyn :redefs") | (assert (= 1 (dynamicdef2-inc)) "before redefinition with dyn :redefs") | ||||||
| (def dynamicdef2 1) | (def dynamicdef2 1) | ||||||
| (assert (= 2 (dynamicdef2-inc)) "after redefinition with dyn :redefs") | (assert (= 2 (dynamicdef2-inc)) "after redefinition with dyn :redefs") | ||||||
| (setdyn :redefs nil) | (setdyn :redef nil) | ||||||
|  |  | ||||||
| # Denormal tables and structs | # Denormal tables and structs | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Calvin Rose
					Calvin Rose