shithub: sl

Download patch

ref: b578f6fc6c2d7b99c42e6477e0fe8a86a0666e03
parent: f2a60067b610cdc82921bcfa4e0a6f198af7497e
author: Sigrid Solveig Haflínudóttir <sigrid@ftrv.se>
date: Fri Jan 24 00:42:01 EST 2025

an assortment of function attributes

--- a/3rd/brieflz/brieflz.c
+++ b/3rd/brieflz/brieflz.c
@@ -27,12 +27,10 @@
 
 #include "brieflz.h"
 
-#if defined(__clang__) && defined(__has_builtin)
-#  if __has_builtin(__builtin_clz)
+#if defined(__has_builtin)
+#  if __has_builtin(__builtin_clzl)
 #    define BLZ_BUILTIN_GCC
 #  endif
-#elif __GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 4)
-#  define BLZ_BUILTIN_GCC
 #endif
 
 // Type used to store values in workmem.
@@ -211,6 +209,7 @@
 };
 #endif
 
+fl_constfn
 static int
 blz_log2(unsigned long n)
 {
@@ -233,6 +232,7 @@
 #endif
 }
 
+fl_constfn
 static unsigned long
 blz_gamma_cost(unsigned long n)
 {
@@ -241,6 +241,7 @@
 	return 2 * (unsigned long) blz_log2(n);
 }
 
+fl_constfn
 static unsigned long
 blz_match_cost(unsigned long pos, unsigned long len)
 {
@@ -411,6 +412,7 @@
 // This is Fibonacci hashing, also known as Knuth's multiplicative hash. The
 // constant is a prime close to 2^32/phi.
 //
+fl_purefn
 static unsigned long
 blz_hash4_bits(const unsigned char *p, int bits)
 {
@@ -424,6 +426,7 @@
 	return (val * 2654435761U) >> (32 - bits);
 }
 
+fl_purefn
 static unsigned long
 blz_hash4(const unsigned char *p)
 {
--- a/3rd/brieflz/brieflz_hashbucket.h
+++ b/3rd/brieflz/brieflz_hashbucket.h
@@ -28,6 +28,7 @@
 #ifndef BRIEFLZ_HASHBUCKET_H_INCLUDED
 #define BRIEFLZ_HASHBUCKET_H_INCLUDED
 
+fl_constfn
 static size_t
 blz_hashbucket_workmem_size(size_t src_size, unsigned int bucket_size)
 {
--- a/3rd/dlmalloc.inc
+++ b/3rd/dlmalloc.inc
@@ -2707,6 +2707,7 @@
   ((char*)(A) >= S->base && (char*)(A) < S->base + S->size)
 
 /* Return segment holding given address */
+fl_purefn
 static msegmentptr segment_holding(mstate m, char* addr) {
   msegmentptr sp = &m->seg;
   for (;;) {
@@ -2718,6 +2719,7 @@
 }
 
 /* Return true if segment contains a segment link */
+fl_purefn
 static int has_segment_link(mstate m, msegmentptr ss) {
   msegmentptr sp = &m->seg;
   for (;;) {
@@ -5350,14 +5352,17 @@
   return result;
 }
 
+fl_purefn
 size_t dlmalloc_footprint(void) {
   return gm->footprint;
 }
 
+fl_purefn
 size_t dlmalloc_max_footprint(void) {
   return gm->max_footprint;
 }
 
+fl_purefn
 size_t dlmalloc_footprint_limit(void) {
   size_t maf = gm->footprint_limit;
   return maf == 0 ? MAX_SIZE_T : maf;
@@ -5390,6 +5395,7 @@
   return change_mparam(param_number, value);
 }
 
+fl_purefn
 size_t dlmalloc_usable_size(void* mem) {
   if (mem != 0) {
     mchunkptr p = mem2chunk(mem);
--- a/3rd/fn.h
+++ b/3rd/fn.h
@@ -61,8 +61,11 @@
 	} \
 	struct dummy
 
+fl_purefn
 Tcheck_get(Trie *,       Tbranch, twigs, t->ptr);
+fl_purefn
 Tcheck_get(const char *, Tleaf,   key,   t->ptr);
+fl_purefn
 Tcheck_get(void *,       Tleaf,   val,   (void*)(uintptr_t)t->index);
 
 // index word layout
--- a/3rd/mp/mp.h
+++ b/3rd/mp/mp.h
@@ -76,15 +76,15 @@
 mpint*	betomp(uint8_t*, uint32_t, mpint*);	/* byte array, big-endian */
 int	mptobe(mpint*, uint8_t*, uint32_t, uint8_t**);
 void	mptober(mpint *b, uint8_t *p, int n);
-uint32_t	mptoui(mpint*);			/* unsigned int */
+uint32_t	mptoui(mpint*) fl_purefn;			/* unsigned int */
 mpint*	uitomp(uint32_t, mpint*);
-int	mptoi(mpint*);			/* int */
+int	mptoi(mpint*) fl_purefn;			/* int */
 mpint*	itomp(int, mpint*);
-uint64_t	mptouv(mpint*);			/* unsigned int64_t */
+uint64_t	mptouv(mpint*) fl_purefn;			/* unsigned int64_t */
 mpint*	uvtomp(uint64_t, mpint*);
-int64_t	mptov(mpint*);			/* int64_t */
+int64_t	mptov(mpint*) fl_purefn;			/* int64_t */
 mpint*	vtomp(int64_t, mpint*);
-double	mptod(mpint*);			/* double */
+double	mptod(mpint*) fl_purefn;			/* double */
 mpint*	dtomp(double, mpint*);
 
 /* divide the 2 digit dividend by the one digit divisor and stick in quotient */
@@ -118,7 +118,7 @@
 void	mpdiv(mpint *dividend, mpint *divisor,  mpint *quotient, mpint *remainder);
 
 /* return neg, 0, pos as b1-b2 is neg, 0, pos */
-int	mpcmp(mpint *b1, mpint *b2);
+int	mpcmp(mpint *b1, mpint *b2) fl_purefn;
 
 /* res = s != 0 ? b1 : b2 */
 void	mpsel(int s, mpint *b1, mpint *b2, mpint *res);
@@ -130,7 +130,7 @@
 void	mpinvert(mpint *b, mpint *m, mpint *res);
 
 /* bit counting */
-uint32_t	mpsignif(mpint*);	/* number of sigificant bits in mantissa */
+uint32_t	mpsignif(mpint*) fl_purefn;	/* number of sigificant bits in mantissa */
 uint32_t	mplowbits0(mpint*);	/* k, where n = 2**k * q for odd q */
 
 /* well known constants */
@@ -158,11 +158,11 @@
 void	mpvectsmul(mpdigit *a, int alen, mpdigit *b, int blen, mpdigit *p);
 
 /* sign of a - b or zero if the same */
-int	mpveccmp(mpdigit *a, int alen, mpdigit *b, int blen);
-int	mpvectscmp(mpdigit *a, int alen, mpdigit *b, int blen);
+int	mpveccmp(mpdigit *a, int alen, mpdigit *b, int blen) fl_purefn;
+int	mpvectscmp(mpdigit *a, int alen, mpdigit *b, int blen) fl_purefn;
 
 /* playing with magnitudes */
-int	mpmagcmp(mpint *b1, mpint *b2);
+int	mpmagcmp(mpint *b1, mpint *b2) fl_purefn;
 void	mpmagadd(mpint *b1, mpint *b2, mpint *sum);	/* sum = b1+b2 */
 void	mpmagsub(mpint *b1, mpint *b2, mpint *sum);	/* sum = b1+b2 */
 
--- a/3rd/mp/test/fns.h
+++ b/3rd/mp/test/fns.h
@@ -1,5 +1,5 @@
 ldint* ldnew(int);
-int ldcmp(ldint *, ldint *);
+int ldcmp(ldint *, ldint *) fl_purefn;
 int ldmagcmp(ldint *, ldint *);
 void ldadd(ldint *, ldint *, ldint *);
 void ldsub(ldint *, ldint *, ldint *);
@@ -16,7 +16,7 @@
 void lddiv_(ldint *, ldint *, ldint *, ldint *);
 void ldfree(ldint *);
 void testgen(int, ldint *);
-int ldmpeq(ldint *, mpint *);
+int ldmpeq(ldint *, mpint *) fl_purefn;
 mpint* ldtomp(ldint *, mpint *);
 void mptarget(mpint *);
 void tests(void);
--- a/3rd/spooky.c
+++ b/3rd/spooky.c
@@ -64,6 +64,7 @@
 //
 // Read uint64_t in little-endian order.
 //
+fl_purefn
 static inline uint64_t
 spooky_read_le64(const uint64_t *s)
 {
--- a/3rd/utf/runeistypedata
+++ b/3rd/utf/runeistypedata
@@ -1,4 +1,4 @@
-static
+static const
 uint8_t	_mergeddata[11769] =
 {
 	0,0,0,0,0,0,0,0,0,1,1,1,1,1,0,0,
@@ -738,7 +738,7 @@
 	2,2,2,2,2,2,2,2,2,0,0,0,0,0,0,0,
 	0,0,0,0,0,0,0,0,0,
 };
-static
+static const
 uint16_t	_mergedidx2[3755] =
 {
 	0,14,30,46,61,66,81,86,97,103,119,130,146,155,170,179,
@@ -977,7 +977,7 @@
 	11753,11753,11753,11753,11753,11753,11753,11753,11753,11753,11753,11753,11753,11753,11753,11753,
 	11753,11753,11753,11753,11753,11753,11753,11753,11753,11753,11753,
 };
-static
+static const
 uint16_t	_mergedidx1[2048] =
 {
 	0,64,128,192,256,320,383,447,511,564,579,643,707,771,771,771,
--- a/3rd/utf/utf.h
+++ b/3rd/utf/utf.h
@@ -11,9 +11,9 @@
 
 int chartorune(Rune *rune, const char *str);
 int runetochar(char *str, const Rune *rune);
-int runenlen(const Rune *r, int nrune);
-int fullrune(const char *str, int n);
-int runelen(Rune c);
+int runenlen(const Rune *r, int nrune) fl_purefn;
+int fullrune(const char *str, int n) fl_purefn;
+int runelen(Rune c) fl_constfn;
 Rune tolowerrune(Rune c) fl_constfn;
 Rune toupperrune(Rune c) fl_constfn;
 Rune totitlerune(Rune c) fl_constfn;
--- a/3rd/wcwidth.c
+++ b/3rd/wcwidth.c
@@ -25,7 +25,7 @@
 // From https://github.com/jquast/wcwidth/blob/master/wcwidth/table_zero.py
 // from https://github.com/jquast/wcwidth/pull/64
 // at commit 1b9b6585b0080ea5cb88dc9815796505724793fe (2022-12-16):
-static struct width_interval ZERO_WIDTH[] = {
+static const struct width_interval ZERO_WIDTH[] = {
 	{0x00300, 0x0036f},  // Combining Grave Accent  ..Combining Latin Small Le
 	{0x00483, 0x00489},  // Combining Cyrillic Titlo..Combining Cyrillic Milli
 	{0x00591, 0x005bd},  // Hebrew Accent Etnahta   ..Hebrew Point Meteg
@@ -374,7 +374,7 @@
 // https://github.com/jquast/wcwidth/blob/master/wcwidth/table_wide.py
 // from https://github.com/jquast/wcwidth/pull/64
 // at commit 1b9b6585b0080ea5cb88dc9815796505724793fe (2022-12-16):
-static struct width_interval WIDE_EASTASIAN[] = {
+static const struct width_interval WIDE_EASTASIAN[] = {
 	{0x01100, 0x0115f},  // Hangul Choseong Kiyeok  ..Hangul Choseong Filler
 	{0x0231a, 0x0231b},  // Watch		   ..Hourglass
 	{0x02329, 0x0232a},  // Left-pointing Angle Brac..Right-pointing Angle Bra
@@ -498,8 +498,9 @@
 	{0x30000, 0x3fffd},  // Cjk Unified Ideograph-30..(nil)
 };
 
+fl_constfn
 static bool
-intable(struct width_interval* table, int table_length, int c)
+intable(const struct width_interval* table, int table_length, int c)
 {
 	// First quick check for Latin1 etc. characters.
 	if(c < table[0].start)
--- a/builtins.c
+++ b/builtins.c
@@ -51,6 +51,7 @@
 	return first;
 }
 
+fl_purefn
 BUILTIN("assq", assq)
 {
 	argcount(nargs, 2);
@@ -68,6 +69,7 @@
 	return FL_f;
 }
 
+fl_purefn
 BUILTIN("memq", memq)
 {
 	argcount(nargs, 2);
@@ -130,6 +132,7 @@
 	type_error("sequence", a);
 }
 
+_Noreturn
 BUILTIN("raise", raise)
 {
 	argcount(nargs, 1);
@@ -136,6 +139,7 @@
 	fl_raise(args[0]);
 }
 
+_Noreturn
 BUILTIN("exit", exit)
 {
 	if(nargs > 1)
@@ -151,6 +155,7 @@
 	return symbol(cvalue_data(args[0]), true);
 }
 
+fl_purefn
 BUILTIN("keyword?", keywordp)
 {
 	argcount(nargs, 1);
@@ -158,6 +163,7 @@
 			iskeyword((symbol_t*)ptr(args[0]))) ? FL_t : FL_f;
 }
 
+fl_purefn
 BUILTIN("top-level-value", top_level_value)
 {
 	argcount(nargs, 1);
@@ -201,6 +207,7 @@
 	return lst;
 }
 
+fl_purefn
 BUILTIN("constant?", constantp)
 {
 	argcount(nargs, 1);
@@ -214,6 +221,7 @@
 	return FL_t;
 }
 
+fl_purefn
 BUILTIN("integer-valued?", integer_valuedp)
 {
 	argcount(nargs, 1);
@@ -243,6 +251,7 @@
 	return FL_f;
 }
 
+fl_purefn
 BUILTIN("integer?", integerp)
 {
 	argcount(nargs, 1);
@@ -252,6 +261,7 @@
 		FL_t : FL_f;
 }
 
+fl_purefn
 BUILTIN("bignum?", bignump)
 {
 	argcount(nargs, 1);
--- a/cvalues.c
+++ b/cvalues.c
@@ -578,6 +578,7 @@
 	return size_wrap(n);
 }
 
+fl_purefn
 BUILTIN("typeof", typeof)
 {
 	argcount(nargs, 1);
@@ -666,6 +667,7 @@
 	return cvalue_copy(args[0]);
 }
 
+fl_purefn
 BUILTIN("plain-old-data?", plain_old_datap)
 {
 	argcount(nargs, 1);
@@ -795,6 +797,7 @@
 	return args[2];
 }
 
+fl_purefn
 BUILTIN("builtin", builtin)
 {
 	argcount(nargs, 1);
--- a/dos/platform.h
+++ b/dos/platform.h
@@ -51,4 +51,4 @@
 #include "mp.h"
 #include "utf.h"
 
-int wcwidth(Rune c);
+int wcwidth(Rune c) fl_constfn;
--- a/flisp.c
+++ b/flisp.c
@@ -245,6 +245,7 @@
 	return fn_builtin_gensym(nil, 0);
 }
 
+fl_purefn
 BUILTIN("gensym?", gensymp)
 {
 	argcount(nargs, 1);
@@ -1063,6 +1064,7 @@
 	return fv;
 }
 
+fl_purefn
 BUILTIN("function:code", function_code)
 {
 	argcount(nargs, 1);
@@ -1072,6 +1074,7 @@
 	return fn_bcode(v);
 }
 
+fl_purefn
 BUILTIN("function:vals", function_vals)
 {
 	argcount(nargs, 1);
@@ -1081,6 +1084,7 @@
 	return fn_vals(v);
 }
 
+fl_purefn
 BUILTIN("function:env", function_env)
 {
 	argcount(nargs, 1);
--- a/iostream.c
+++ b/iostream.c
@@ -47,6 +47,7 @@
 	return iscvalue(v) && cv_class(ptr(v)) == FL(iostreamtype);
 }
 
+fl_purefn
 BUILTIN("iostream?", iostreamp)
 {
 	argcount(nargs, 1);
@@ -53,6 +54,7 @@
 	return isiostream(args[0]) ? FL_t : FL_f;
 }
 
+fl_purefn
 BUILTIN("eof-object?", eof_objectp)
 {
 	argcount(nargs, 1);
@@ -59,6 +61,7 @@
 	return args[0] == FL_eof ? FL_t : FL_f;
 }
 
+fl_purefn
 ios_t *
 toiostream(value_t v)
 {
@@ -196,6 +199,7 @@
 	return FL_void;
 }
 
+fl_purefn
 BUILTIN("io-eof?", io_eofp)
 {
 	argcount(nargs, 1);
--- a/maxstack.inc
+++ b/maxstack.inc
@@ -1,3 +1,4 @@
+fl_purefn
 static int
 compute_maxstack(uint8_t *code, size_t len)
 {
--- a/meson.build
+++ b/meson.build
@@ -16,12 +16,8 @@
 	#'-Wcast-qual',
 	#'-Wconversion',
 	#'-Wfloat-equal',
-	#'-Wmissing-noreturn',
 	#'-Wsign-conversion',
-	#'-Wsuggest-attribute=const',
 	#'-Wsuggest-attribute=malloc',
-	#'-Wsuggest-attribute=noreturn',
-	#'-Wsuggest-attribute=pure',
 	#'-Wsuggest-attribute=returns_nonnull',
 	'-Waggregate-return',
 	'-Werror=odr',
@@ -28,6 +24,7 @@
 	'-Werror=strict-aliasing',
 	'-Wformat=2',
 	'-Wint-to-pointer-cast',
+	'-Wmissing-noreturn',
 	'-Wmissing-prototypes',
 	'-Wno-format-y2k',
 	'-Wpointer-arith',
@@ -60,6 +57,9 @@
 		'-Wlogical-op',
 		'-Wunused-const-variable=2',
 		'-Werror=lto-type-mismatch',
+		'-Wsuggest-attribute=const',
+		'-Wsuggest-attribute=noreturn',
+		'-Wsuggest-attribute=pure',
 		language: 'c',
 	)
 endif
--- a/operators.c
+++ b/operators.c
@@ -20,6 +20,7 @@
 	return mpzero;
 }
 
+fl_purefn
 double
 conv_to_double(void *data, numerictype_t tag)
 {
@@ -68,6 +69,7 @@
 
 // FIXME sign with mpint
 #define CONV_TO_INTTYPE(name, ctype) \
+fl_purefn \
 ctype \
 conv_to_##name(void *data, numerictype_t tag) \
 { \
@@ -94,6 +96,7 @@
 // this is needed to work around an UB casting negative
 // floats and doubles to uint64. you need to cast to int64
 // first.
+fl_purefn
 uint64_t
 conv_to_uint64(void *data, numerictype_t tag)
 {
@@ -122,6 +125,7 @@
 	return 0;
 }
 
+fl_purefn
 int
 cmp_same_lt(void *a, void *b, numerictype_t tag)
 {
@@ -141,6 +145,7 @@
 	return 0;
 }
 
+fl_purefn
 int
 cmp_same_eq(void *a, void *b, numerictype_t tag)
 {
--- a/string.c
+++ b/string.c
@@ -9,6 +9,7 @@
 #include "equal.h"
 #include "iostream.h"
 
+fl_purefn
 BUILTIN("string?", stringp)
 {
 	argcount(nargs, 1);
@@ -243,6 +244,7 @@
 	return mk_rune(totitlerune(*(Rune*)cp_data(cp)));
 }
 
+fl_purefn
 BUILTIN("char-alphabetic?", char_alphabeticp)
 {
 	argcount(nargs, 1);
@@ -252,6 +254,7 @@
 	return isalpharune(*(Rune*)cp_data(cp)) ? FL_t : FL_f;
 }
 
+fl_purefn
 BUILTIN("char-lower-case?", char_lower_casep)
 {
 	argcount(nargs, 1);
@@ -261,6 +264,7 @@
 	return islowerrune(*(Rune*)cp_data(cp)) ? FL_t : FL_f;
 }
 
+fl_purefn
 BUILTIN("char-upper-case?", char_upper_casep)
 {
 	argcount(nargs, 1);
@@ -270,6 +274,7 @@
 	return isupperrune(*(Rune*)cp_data(cp)) ? FL_t : FL_f;
 }
 
+fl_purefn
 BUILTIN("char-title-case?", char_title_casep)
 {
 	argcount(nargs, 1);
@@ -279,6 +284,7 @@
 	return istitlerune(*(Rune*)cp_data(cp)) ? FL_t : FL_f;
 }
 
+fl_purefn
 BUILTIN("char-numeric?", char_numericp)
 {
 	argcount(nargs, 1);
@@ -288,6 +294,7 @@
 	return isdigitrune(*(Rune*)cp_data(cp)) ? FL_t : FL_f;
 }
 
+fl_purefn
 BUILTIN("char-whitespace?", char_whitespacep)
 {
 	argcount(nargs, 1);
@@ -392,6 +399,7 @@
 	return n;
 }
 
+fl_purefn
 BUILTIN("string-utf8?", string_utf8p)
 {
 	argcount(nargs, 1);
--- a/table.c
+++ b/table.c
@@ -74,6 +74,7 @@
 	return iscvalue(v) && cv_class(ptr(v)) == FL(tabletype);
 }
 
+fl_purefn
 BUILTIN("table?", tablep)
 {
 	argcount(nargs, 1);
@@ -133,6 +134,7 @@
 	return args[0];
 }
 
+_Noreturn
 static void
 key_error(value_t key)
 {
@@ -140,6 +142,7 @@
 }
 
 // (get table key [default])
+fl_purefn
 BUILTIN("get", get)
 {
 	if(nargs != 3)
@@ -155,6 +158,7 @@
 }
 
 // (has? table key)
+fl_purefn
 BUILTIN("has?", has)
 {
 	argcount(nargs, 2);