You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

565 lines
19 KiB

25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
25 years ago
  1. // General object definitions: pointers, reference counting, garbage collection.
  2. #ifndef _CL_OBJECT_H
  3. #define _CL_OBJECT_H
  4. #include "cln/types.h"
  5. #include "cln/modules.h"
  6. #include <cstdlib>
  7. namespace cln {
  8. // We don't have to deal with circular structures, so normal reference counting
  9. // is sufficient. Is also has the advantage of being mostly non-interrupting.
  10. // An object is either a pointer to heap allocated data
  11. // or immediate data.
  12. // It is possible to distinguish these because pointers are aligned.
  13. // cl_uint_alignment is the guaranteed alignment of a `void*' or `long'
  14. // in memory. Must be > 1.
  15. #if defined(__m68k__)
  16. #define cl_word_alignment 2
  17. #endif
  18. #if defined(__i386__) || defined(__mips__) || defined(__mipsel__) || defined(__sparc__) || defined(__hppa__) || defined(__arm__) || defined(__rs6000__) || defined(__m88k__) || defined(__convex__) || defined(__s390__)
  19. #define cl_word_alignment 4
  20. #endif
  21. #if defined(__alpha__) || defined(__mips64__) || defined(__sparc64__) || defined(__ia64__) || defined(__x86_64__)
  22. #define cl_word_alignment 8
  23. #endif
  24. #if !defined(cl_word_alignment)
  25. #error "Define cl_word_alignment for your CPU!"
  26. #endif
  27. // Four basic classes are introduced:
  28. //
  29. // gcobject rcobject
  30. //
  31. // gcpointer rcpointer
  32. //
  33. // `gcobject' = garbage collectible object (pointer or immediate),
  34. // `gcpointer' = garbage collectible pointer,
  35. // `rcobject' = reference counted object (pointer or immediate),
  36. // `rcpointer' = reference counted pointer.
  37. //
  38. // "garbage collectible" means that a reference count is maintained, and
  39. // when the reference count drops to 0, the object is freed. This is useful
  40. // for all kind of short- or long-lived objects.
  41. // "reference counted" means that a reference count is maintained, which
  42. // cannot drop to 0. This is useful for objects which are registered in a
  43. // global cache table, in order to know which objects can be thrown away
  44. // when the cache is cleaned. (If the cache were never cleaned, its objects
  45. // would never be freed, and we could get away with normal C pointers.)
  46. //
  47. // It is permissible to treat a `rcobject' as a `gcobject', and a `rcpointer'
  48. // as a `gcpointer', but this just increases the destructor and copy-constructor
  49. // overhead.
  50. // It is also permissible to treat a `gcpointer' as a `gcobject', and a
  51. // `rcpointer' as a `rcobject', but this just increases the destructor and
  52. // copy-constructor overhead.
  53. // Immediate data is a word, as wide as a pointer.
  54. typedef sintP cl_sint;
  55. typedef uintP cl_uint; // This ought to be called `cl_word'.
  56. #define cl_pointer_size intPsize
  57. // NB: (cl_pointer_size==64) implies defined(HAVE_FAST_LONGLONG)
  58. #if (cl_pointer_size==64)
  59. #define CL_WIDE_POINTERS
  60. #endif
  61. // Distinguish immediate data from pointers.
  62. inline cl_boolean cl_pointer_p (cl_uint word)
  63. {
  64. return (cl_boolean)((word & (cl_word_alignment-1)) == 0);
  65. }
  66. inline cl_boolean cl_immediate_p (cl_uint word)
  67. {
  68. return (cl_boolean)((word & (cl_word_alignment-1)) != 0);
  69. }
  70. // Immediate data: Fixnum, Short Float, maybe Single Float.
  71. // They have type tags.
  72. // |...............................|......|
  73. // cl_value cl_tag
  74. // Number of bits reserved for tagging information:
  75. #if (cl_word_alignment <= 4)
  76. #define cl_tag_len 2
  77. #else
  78. #define cl_tag_len 3
  79. #endif
  80. #define cl_tag_shift 0
  81. #if (cl_pointer_size == 64)
  82. #define cl_value_shift 32
  83. #else
  84. #define cl_value_shift (cl_tag_len+cl_tag_shift)
  85. #endif
  86. #define cl_value_len (cl_pointer_size - cl_value_shift)
  87. #define cl_tag_mask (((1UL << cl_tag_len) - 1) << cl_tag_shift)
  88. #define cl_value_mask (((1UL << cl_value_len) - 1) << cl_value_shift)
  89. // Return the tag of a word.
  90. inline cl_uint cl_tag (cl_uint word)
  91. {
  92. return (word & cl_tag_mask) >> cl_tag_shift;
  93. }
  94. // Return the value (unsigned) of a word.
  95. inline cl_uint cl_value (cl_uint word)
  96. {
  97. // This assumes cl_value_shift + cl_value_len == cl_pointer_size.
  98. return word >> cl_value_shift;
  99. }
  100. // Return a word, combining a value and a tag.
  101. inline cl_uint cl_combine (cl_uint tag, cl_uint value)
  102. {
  103. return (value << cl_value_shift) + (tag << cl_tag_shift);
  104. }
  105. inline cl_uint cl_combine (cl_uint tag, cl_sint value)
  106. {
  107. // This assumes cl_value_shift + cl_value_len == cl_pointer_size.
  108. return (value << cl_value_shift) + (tag << cl_tag_shift);
  109. }
  110. // Keep the compiler happy.
  111. inline cl_uint cl_combine (cl_uint tag, unsigned int value)
  112. { return cl_combine(tag, (cl_uint)value); }
  113. inline cl_uint cl_combine (cl_uint tag, int value)
  114. { return cl_combine(tag, (cl_sint)value); }
  115. #ifdef HAVE_LONGLONG
  116. inline cl_uint cl_combine (cl_uint tag, unsigned long long value)
  117. { return cl_combine(tag, (cl_uint)value); }
  118. inline cl_uint cl_combine (cl_uint tag, long long value)
  119. { return cl_combine(tag, (cl_uint)value); }
  120. #endif
  121. // Definition of the tags.
  122. #if !defined(CL_WIDE_POINTERS)
  123. #if (cl_word_alignment == 2)
  124. #define cl_FN_tag 1
  125. #define cl_SF_tag 3 // must satisfy the cl_immediate_p predicate!
  126. #endif
  127. #if (cl_word_alignment == 4)
  128. #define cl_FN_tag 1
  129. #define cl_SF_tag 2
  130. #endif
  131. #else // CL_WIDE_POINTERS
  132. // Single Floats are immediate as well.
  133. #define cl_FN_tag 1
  134. #define cl_SF_tag 2
  135. #define cl_FF_tag 3
  136. #endif
  137. // Corresponding classes.
  138. extern const struct cl_class * cl_immediate_classes [1<<cl_tag_len];
  139. // Heap allocated data contains a header, for two purposes:
  140. // - dynamic typing,
  141. // - reference count (a portable alternative to garbage collection,
  142. // or the basis for a portable and interoperable garbage collection).
  143. struct cl_heap {
  144. int refcount; // reference count
  145. const struct cl_class * type; // type tag
  146. };
  147. // Function to destroy the contents of a heap object.
  148. typedef void (*cl_heap_destructor_function) (cl_heap* pointer);
  149. // Flags, may be ORed together.
  150. #define cl_class_flags_subclass_complex 1 // all instances belong to cl_N
  151. #define cl_class_flags_subclass_real 2 // all instances belong to cl_R
  152. #define cl_class_flags_subclass_float 4 // all instances belong to cl_F
  153. #define cl_class_flags_subclass_rational 8 // all instances belong to cl_RA
  154. #define cl_class_flags_number_ring 16 // all instances are rings whose
  155. // elements belong to cl_number
  156. #define cl_class_flags_modint_ring 32 // all instances are rings whose
  157. // elements belong to cl_MI
  158. // Function to print an object for debugging, to cerr.
  159. typedef void (*cl_heap_dprint_function) (cl_heap* pointer);
  160. struct cl_class {
  161. cl_heap_destructor_function destruct;
  162. int flags;
  163. cl_heap_dprint_function dprint;
  164. };
  165. // Free an object on heap.
  166. extern void cl_free_heap_object (cl_heap* pointer);
  167. // Debugging support for dynamic typing: Register a debugging print function.
  168. #define cl_register_type_printer(type,printer) \
  169. { extern cl_class type; type.dprint = (printer); }
  170. // cl_private_thing: An immediate value or a pointer into the heap.
  171. // This must be as wide as a `cl_uint'.
  172. // (Actually, this ought to be a union { void*; cl_uint; }, but using
  173. // a pointer type generates better code.)
  174. // Never throw away a cl_private_thing, or reference counts will be wrong!
  175. typedef struct cl_anything * cl_private_thing;
  176. // Increment the reference count.
  177. inline void cl_inc_pointer_refcount (cl_heap* pointer)
  178. {
  179. pointer->refcount++;
  180. }
  181. // Decrement the reference count of a garbage collected pointer.
  182. inline void cl_gc_dec_pointer_refcount (cl_heap* pointer)
  183. {
  184. if (--pointer->refcount == 0)
  185. cl_free_heap_object(pointer);
  186. }
  187. // Decrement the reference count of a reference counted pointer.
  188. inline void cl_rc_dec_pointer_refcount (cl_heap* pointer)
  189. {
  190. --pointer->refcount;
  191. }
  192. // Increment the reference count.
  193. // This must be a macro, not an inline function, because pointer_p() and
  194. // inc_pointer_refcount() are non-virtual member functions, so that the
  195. // compiler can optimize it.
  196. #define cl_inc_refcount(x) \
  197. if ((x).pointer_p()) \
  198. (x).inc_pointer_refcount(); \
  199. // Decrement the reference count.
  200. // This must be a macro, not an inline function, because pointer_p() and
  201. // dec_pointer_refcount() are non-virtual member functions, so that the
  202. // compiler can optimize it.
  203. #define cl_dec_refcount(x) \
  204. if ((x).pointer_p()) \
  205. (x).dec_pointer_refcount(); \
  206. // The declaration of a copy constructor.
  207. // Restriction: The base class's default constructor must do nothing or
  208. // initialize `pointer' to a constant expression.
  209. #define CL_DEFINE_COPY_CONSTRUCTOR1(_class_) \
  210. _CL_DEFINE_COPY_CONSTRUCTOR1(_class_,_class_)
  211. #define _CL_DEFINE_COPY_CONSTRUCTOR1(_class_,_classname_) \
  212. inline _class_::_classname_ (const _class_& x) \
  213. { \
  214. cl_uint x_word = x.word; \
  215. cl_inc_refcount(x); \
  216. this->word = x_word; \
  217. }
  218. // The declaration of a copy constructor.
  219. // Restriction: The base class must have the usual `cl_private_thing'
  220. // constructor. Drawback: The base class must be known here.
  221. #define CL_DEFINE_COPY_CONSTRUCTOR2(_class_,_baseclass_) \
  222. _CL_DEFINE_COPY_CONSTRUCTOR2(_class_,_class_,_baseclass_)
  223. #define _CL_DEFINE_COPY_CONSTRUCTOR2(_class_,_classname_,_baseclass_) \
  224. inline _class_::_classname_ (const _class_& x) \
  225. : _baseclass_ (as_cl_private_thing(x)) {}
  226. // The declaration of an assignment operator.
  227. #define CL_DEFINE_ASSIGNMENT_OPERATOR(dest_class,src_class) \
  228. inline dest_class& dest_class::operator= (const src_class& x) \
  229. { \
  230. /* Be careful, we might be assigning x to itself. */ \
  231. cl_uint x_word = x.word; \
  232. cl_inc_refcount(x); \
  233. cl_dec_refcount(*this); \
  234. this->word = x_word; \
  235. return *this; \
  236. }
  237. // We have a small problem with destructors: The specialized destructor
  238. // of a leaf class such as `cl_SF' should be more efficient than the
  239. // general destructor for `cl_N'. Since (by C++ specs) destructing a cl_SF
  240. // would run the destructors for cl_SF, cl_F, cl_R, cl_N (in that order),
  241. // and in the last step the compiler does not know any more that the object
  242. // actually is a cl_SF, there is no way to optimize the destructor!
  243. // ("progn-reversed" method combination is evil.)
  244. // And if we define "mirror"/"shadow" classes with no destructors (such
  245. // that `cl_F' inherits from `cl_F_no_destructor' buts adds a destructor)
  246. // then we need to add explicit conversion operators cl_SF -> cl_F -> cl_R ...,
  247. // with the effect that calling an overloaded function like `as_cl_F'
  248. // (which has two signatures `as_cl_F(cl_number)' and `as_cl_F(cl_F)')
  249. // with a cl_SF argument gives an "call of overloaded function is ambiguous"
  250. // error.
  251. // There is no help: If we want overloaded functions to be callable in a way
  252. // that makes sense, `cl_SF' has to be a subclass of `cl_F', and then the
  253. // destructor of `cl_SF' will do at least as much computation as the `cl_F'
  254. // destructor. Praise C++ ! :-((
  255. // (Even making `pointer_p()' a virtual function would not help.)
  256. // This is obnoxious.
  257. template <class key1_type, class value_type> struct cl_htentry1;
  258. // The four concrete classes of all objects.
  259. class cl_gcobject {
  260. public: /* ugh */
  261. union {
  262. void* pointer;
  263. cl_heap* heappointer;
  264. cl_uint word;
  265. };
  266. public:
  267. // Default constructor. (Used for objects with no initializer.)
  268. cl_gcobject ();
  269. // Destructor. (Used when a variable goes out of scope.)
  270. ~cl_gcobject ();
  271. // Copy constructor.
  272. cl_gcobject (const cl_gcobject&);
  273. // Assignment operator.
  274. cl_gcobject& operator= (const cl_gcobject&);
  275. // Distinguish immediate data from pointer.
  276. cl_boolean pointer_p() const
  277. { return cl_pointer_p(word); }
  278. // Reference counting.
  279. void inc_pointer_refcount () const
  280. { cl_inc_pointer_refcount(heappointer); }
  281. void dec_pointer_refcount () const
  282. { cl_gc_dec_pointer_refcount(heappointer); }
  283. // Return the type tag of an immediate number.
  284. cl_uint nonpointer_tag () const
  285. { return cl_tag(word); }
  286. // Return the type tag of a heap-allocated number.
  287. const cl_class * pointer_type () const
  288. { return heappointer->type; }
  289. // Private pointer manipulations.
  290. cl_private_thing _as_cl_private_thing () const;
  291. // Private constructor.
  292. cl_gcobject (cl_private_thing p)
  293. #if !(defined(__alpha__) && !defined(__GNUC__))
  294. : pointer (p) {}
  295. #else
  296. { pointer = p; }
  297. #endif
  298. // Debugging output.
  299. void debug_print () const;
  300. // Ability to place an object at a given address.
  301. void* operator new (size_t size, void* ptr) { (void)size; return ptr; }
  302. void* operator new (size_t size) { return ::operator new (size); }
  303. };
  304. inline cl_gcobject::cl_gcobject () {}
  305. inline cl_gcobject::~cl_gcobject () { cl_dec_refcount(*this); }
  306. CL_DEFINE_COPY_CONSTRUCTOR1(cl_gcobject)
  307. CL_DEFINE_ASSIGNMENT_OPERATOR(cl_gcobject,cl_gcobject)
  308. class cl_gcpointer {
  309. public: /* ugh */
  310. union {
  311. void* pointer;
  312. cl_heap* heappointer;
  313. cl_uint word;
  314. };
  315. public:
  316. // Default constructor. (Used for objects with no initializer.)
  317. cl_gcpointer ();
  318. // Destructor. (Used when a variable goes out of scope.)
  319. ~cl_gcpointer ();
  320. // Copy constructor.
  321. cl_gcpointer (const cl_gcpointer&);
  322. // Assignment operator.
  323. cl_gcpointer& operator= (const cl_gcpointer&);
  324. // Distinguish immediate data from pointer.
  325. cl_boolean pointer_p() const
  326. { return cl_true; }
  327. // Reference counting.
  328. void inc_pointer_refcount () const
  329. { cl_inc_pointer_refcount(heappointer); }
  330. void dec_pointer_refcount () const
  331. { cl_gc_dec_pointer_refcount(heappointer); }
  332. // Return the type tag of an immediate number.
  333. cl_uint nonpointer_tag () const
  334. { return cl_tag(word); }
  335. // Return the type tag of a heap-allocated number.
  336. const cl_class * pointer_type () const
  337. { return heappointer->type; }
  338. // Private pointer manipulations.
  339. cl_private_thing _as_cl_private_thing () const;
  340. // Private constructor.
  341. cl_gcpointer (cl_private_thing p)
  342. #if !(defined(__alpha__) && !defined(__GNUC__))
  343. : pointer (p) {}
  344. #else
  345. { pointer = p; }
  346. #endif
  347. // Debugging output.
  348. void debug_print () const;
  349. // Ability to place an object at a given address.
  350. void* operator new (size_t size, void* ptr) { (void)size; return ptr; }
  351. void* operator new (size_t size) { return ::operator new (size); }
  352. };
  353. inline cl_gcpointer::cl_gcpointer () {}
  354. inline cl_gcpointer::~cl_gcpointer () { cl_dec_refcount(*this); }
  355. CL_DEFINE_COPY_CONSTRUCTOR1(cl_gcpointer)
  356. CL_DEFINE_ASSIGNMENT_OPERATOR(cl_gcpointer,cl_gcpointer)
  357. class cl_rcobject {
  358. public: /* ugh */
  359. union {
  360. void* pointer;
  361. cl_heap* heappointer;
  362. cl_uint word;
  363. };
  364. public:
  365. // Default constructor. (Used for objects with no initializer.)
  366. cl_rcobject ();
  367. // Destructor. (Used when a variable goes out of scope.)
  368. ~cl_rcobject ();
  369. // Copy constructor.
  370. cl_rcobject (const cl_rcobject&);
  371. // Assignment operator.
  372. cl_rcobject& operator= (const cl_rcobject&);
  373. // Distinguish immediate data from pointer.
  374. cl_boolean pointer_p() const
  375. { return cl_pointer_p(word); }
  376. // Reference counting.
  377. void inc_pointer_refcount () const
  378. { cl_inc_pointer_refcount(heappointer); }
  379. void dec_pointer_refcount () const
  380. { cl_rc_dec_pointer_refcount(heappointer); }
  381. // Return the type tag of an immediate number.
  382. cl_uint nonpointer_tag () const
  383. { return cl_tag(word); }
  384. // Return the type tag of a heap-allocated number.
  385. const cl_class * pointer_type () const
  386. { return heappointer->type; }
  387. // Private pointer manipulations.
  388. cl_private_thing _as_cl_private_thing () const;
  389. // Private constructor.
  390. cl_rcobject (cl_private_thing p)
  391. #if !(defined(__alpha__) && !defined(__GNUC__))
  392. : pointer (p) {}
  393. #else
  394. { pointer = p; }
  395. #endif
  396. // Debugging output.
  397. void debug_print () const;
  398. // Ability to place an object at a given address.
  399. void* operator new (size_t size, void* ptr) { (void)size; return ptr; }
  400. void* operator new (size_t size) { return ::operator new (size); }
  401. };
  402. inline cl_rcobject::cl_rcobject () {}
  403. inline cl_rcobject::~cl_rcobject () { cl_dec_refcount(*this); }
  404. CL_DEFINE_COPY_CONSTRUCTOR1(cl_rcobject)
  405. CL_DEFINE_ASSIGNMENT_OPERATOR(cl_rcobject,cl_rcobject)
  406. class cl_rcpointer {
  407. public: /* ugh */
  408. union {
  409. void* pointer;
  410. cl_heap* heappointer;
  411. cl_uint word;
  412. };
  413. public:
  414. // Default constructor. (Used for objects with no initializer.)
  415. cl_rcpointer ();
  416. // Destructor. (Used when a variable goes out of scope.)
  417. ~cl_rcpointer ();
  418. // Copy constructor.
  419. cl_rcpointer (const cl_rcpointer&);
  420. // Assignment operator.
  421. cl_rcpointer& operator= (const cl_rcpointer&);
  422. // Distinguish immediate data from pointer.
  423. cl_boolean pointer_p() const
  424. { return cl_true; }
  425. // Reference counting.
  426. void inc_pointer_refcount () const
  427. { cl_inc_pointer_refcount(heappointer); }
  428. void dec_pointer_refcount () const
  429. { cl_rc_dec_pointer_refcount(heappointer); }
  430. // Return the type tag of an immediate number.
  431. cl_uint nonpointer_tag () const
  432. { return cl_tag(word); }
  433. // Return the type tag of a heap-allocated number.
  434. const cl_class * pointer_type () const
  435. { return heappointer->type; }
  436. // Private pointer manipulations.
  437. cl_private_thing _as_cl_private_thing () const;
  438. // Private constructor.
  439. cl_rcpointer (cl_private_thing p)
  440. #if !(defined(__alpha__) && !defined(__GNUC__))
  441. : pointer (p) {}
  442. #else
  443. { pointer = p; }
  444. #endif
  445. // Debugging output.
  446. void debug_print () const;
  447. // Ability to place an object at a given address.
  448. void* operator new (size_t size, void* ptr) { (void)size; return ptr; }
  449. void* operator new (size_t size) { return ::operator new (size); }
  450. };
  451. inline cl_rcpointer::cl_rcpointer () {}
  452. inline cl_rcpointer::~cl_rcpointer () { cl_dec_refcount(*this); }
  453. CL_DEFINE_COPY_CONSTRUCTOR1(cl_rcpointer)
  454. CL_DEFINE_ASSIGNMENT_OPERATOR(cl_rcpointer,cl_rcpointer)
  455. // Private pointer manipulations.
  456. inline cl_private_thing cl_gcobject::_as_cl_private_thing () const
  457. {
  458. cl_private_thing p = (cl_private_thing) pointer;
  459. cl_inc_refcount(*this);
  460. return p;
  461. }
  462. inline cl_private_thing as_cl_private_thing (const cl_gcobject& x)
  463. {
  464. return x._as_cl_private_thing();
  465. }
  466. inline cl_private_thing cl_gcpointer::_as_cl_private_thing () const
  467. {
  468. cl_private_thing p = (cl_private_thing) pointer;
  469. cl_inc_refcount(*this);
  470. return p;
  471. }
  472. inline cl_private_thing as_cl_private_thing (const cl_gcpointer& x)
  473. {
  474. return x._as_cl_private_thing();
  475. }
  476. inline cl_private_thing cl_rcobject::_as_cl_private_thing () const
  477. {
  478. cl_private_thing p = (cl_private_thing) pointer;
  479. cl_inc_refcount(*this);
  480. return p;
  481. }
  482. inline cl_private_thing as_cl_private_thing (const cl_rcobject& x)
  483. {
  484. return x._as_cl_private_thing();
  485. }
  486. inline cl_private_thing cl_rcpointer::_as_cl_private_thing () const
  487. {
  488. cl_private_thing p = (cl_private_thing) pointer;
  489. cl_inc_refcount(*this);
  490. return p;
  491. }
  492. inline cl_private_thing as_cl_private_thing (const cl_rcpointer& x)
  493. {
  494. return x._as_cl_private_thing();
  495. }
  496. // Note: When we define a function that returns a class object by value,
  497. // we normally return it as const value. The declarations
  498. // T func (...); (A)
  499. // and
  500. // const T func (...); (B)
  501. // behave identically and generate identical code, except that the code
  502. // func(...) = foo;
  503. // compiles fine with (A) but is an error (and yields a warning) with (B).
  504. // We want this warning.
  505. // Define a conversion operator from one object to another object of the
  506. // same size.
  507. #define CL_DEFINE_CONVERTER(target_class) \
  508. operator const target_class & () const \
  509. { \
  510. if (sizeof(*this) != sizeof(target_class)) cl_abort(); \
  511. return * (const target_class *) (void*) this; \
  512. }
  513. } // namespace cln
  514. #endif /* _CL_OBJECT_H */