rubytracking.swg 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. /* -----------------------------------------------------------------------------
  2. * rubytracking.swg
  3. *
  4. * This file contains support for tracking mappings from
  5. * Ruby objects to C++ objects. This functionality is needed
  6. * to implement mark functions for Ruby's mark and sweep
  7. * garbage collector.
  8. * ----------------------------------------------------------------------------- */
  9. #ifdef __cplusplus
  10. extern "C" {
  11. #endif
  12. #if !defined(ST_DATA_T_DEFINED)
  13. /* Needs to be explicitly included for Ruby 1.8 and earlier */
  14. #include <st.h>
  15. #endif
  16. /* Ruby 1.8 actually assumes the first case. */
  17. #if SIZEOF_VOIDP == SIZEOF_LONG
  18. # define SWIG2NUM(v) LONG2NUM((unsigned long)v)
  19. # define NUM2SWIG(x) (unsigned long)NUM2LONG(x)
  20. #elif SIZEOF_VOIDP == SIZEOF_LONG_LONG
  21. # define SWIG2NUM(v) LL2NUM((unsigned long long)v)
  22. # define NUM2SWIG(x) (unsigned long long)NUM2LL(x)
  23. #else
  24. # error sizeof(void*) is not the same as long or long long
  25. #endif
  26. /* Global hash table to store Trackings from C/C++
  27. structs to Ruby Objects.
  28. */
  29. static st_table* swig_ruby_trackings = NULL;
  30. static VALUE swig_ruby_trackings_count(ANYARGS) {
  31. return SWIG2NUM(swig_ruby_trackings->num_entries);
  32. }
  33. /* Setup a hash table to store Trackings */
  34. SWIGRUNTIME void SWIG_RubyInitializeTrackings(void) {
  35. /* Create a hash table to store Trackings from C++
  36. objects to Ruby objects. */
  37. /* Try to see if some other .so has already created a
  38. tracking hash table, which we keep hidden in an instance var
  39. in the SWIG module.
  40. This is done to allow multiple DSOs to share the same
  41. tracking table.
  42. */
  43. VALUE trackings_value = Qnil;
  44. /* change the variable name so that we can mix modules
  45. compiled with older SWIG's - this used to be called "@__trackings__" */
  46. ID trackings_id = rb_intern( "@__safetrackings__" );
  47. VALUE verbose = rb_gv_get("VERBOSE");
  48. rb_gv_set("VERBOSE", Qfalse);
  49. trackings_value = rb_ivar_get( _mSWIG, trackings_id );
  50. rb_gv_set("VERBOSE", verbose);
  51. /* The trick here is that we have to store the hash table
  52. pointer in a Ruby variable. We do not want Ruby's GC to
  53. treat this pointer as a Ruby object, so we convert it to
  54. a Ruby numeric value. */
  55. if (trackings_value == Qnil) {
  56. /* No, it hasn't. Create one ourselves */
  57. swig_ruby_trackings = st_init_numtable();
  58. rb_ivar_set( _mSWIG, trackings_id, SWIG2NUM(swig_ruby_trackings) );
  59. } else {
  60. swig_ruby_trackings = (st_table*)NUM2SWIG(trackings_value);
  61. }
  62. rb_define_virtual_variable("SWIG_TRACKINGS_COUNT", swig_ruby_trackings_count, NULL);
  63. }
  64. /* Add a Tracking from a C/C++ struct to a Ruby object */
  65. SWIGRUNTIME void SWIG_RubyAddTracking(void* ptr, VALUE object) {
  66. /* Store the mapping to the global hash table. */
  67. st_insert(swig_ruby_trackings, (st_data_t)ptr, object);
  68. }
  69. /* Get the Ruby object that owns the specified C/C++ struct */
  70. SWIGRUNTIME VALUE SWIG_RubyInstanceFor(void* ptr) {
  71. /* Now lookup the value stored in the global hash table */
  72. VALUE value;
  73. if (st_lookup(swig_ruby_trackings, (st_data_t)ptr, &value)) {
  74. return value;
  75. } else {
  76. return Qnil;
  77. }
  78. }
  79. /* Remove a Tracking from a C/C++ struct to a Ruby object. It
  80. is very important to remove objects once they are destroyed
  81. since the same memory address may be reused later to create
  82. a new object. */
  83. SWIGRUNTIME void SWIG_RubyRemoveTracking(void* ptr) {
  84. /* Delete the object from the hash table */
  85. st_delete(swig_ruby_trackings, (st_data_t *)&ptr, NULL);
  86. }
  87. /* This is a helper method that unlinks a Ruby object from its
  88. underlying C++ object. This is needed if the lifetime of the
  89. Ruby object is longer than the C++ object */
  90. SWIGRUNTIME void SWIG_RubyUnlinkObjects(void* ptr) {
  91. VALUE object = SWIG_RubyInstanceFor(ptr);
  92. if (object != Qnil) {
  93. if (TYPE(object) != T_DATA)
  94. abort();
  95. DATA_PTR(object) = 0;
  96. }
  97. }
  98. /* This is a helper method that iterates over all the trackings
  99. passing the C++ object pointer and its related Ruby object
  100. to the passed callback function. */
  101. /* Proxy method to abstract the internal trackings datatype */
  102. static int swig_ruby_internal_iterate_callback(void* ptr, VALUE obj, void(*meth)(void* ptr, VALUE obj)) {
  103. (*meth)(ptr, obj);
  104. return ST_CONTINUE;
  105. }
  106. SWIGRUNTIME void SWIG_RubyIterateTrackings( void(*meth)(void* ptr, VALUE obj) ) {
  107. st_foreach(swig_ruby_trackings, (int (*)(ANYARGS))&swig_ruby_internal_iterate_callback, (st_data_t)meth);
  108. }
  109. #ifdef __cplusplus
  110. }
  111. #endif