2000-05-01 13:42:38 +04:00
/**********************************************************************
1998-01-16 15:13:05 +03:00
variable . c -
$ Author $
created at : Tue Apr 19 23 : 55 : 15 JST 1994
* encoding.c: provide basic features for M17N.
* parse.y: encoding aware parsing.
* parse.y (pragma_encoding): encoding specification pragma.
* parse.y (rb_intern3): encoding specified symbols.
* string.c (rb_str_length): length based on characters.
for older behavior, bytesize method added.
* string.c (rb_str_index_m): index based on characters. rindex as
well.
* string.c (succ_char): encoding aware succeeding string.
* string.c (rb_str_reverse): reverse based on characters.
* string.c (rb_str_inspect): encoding aware string description.
* string.c (rb_str_upcase_bang): encoding aware case conversion.
downcase, capitalize, swapcase as well.
* string.c (rb_str_tr_bang): tr based on characters. delete,
squeeze, tr_s, count as well.
* string.c (rb_str_split_m): split based on characters.
* string.c (rb_str_each_line): encoding aware each_line.
* string.c (rb_str_each_char): added. iteration based on
characters.
* string.c (rb_str_strip_bang): encoding aware whitespace
stripping. lstrip, rstrip as well.
* string.c (rb_str_justify): encoding aware justifying (ljust,
rjust, center).
* string.c (str_encoding): get encoding attribute from a string.
* re.c (rb_reg_initialize): encoding aware regular expression
* sprintf.c (rb_str_format): formatting (i.e. length count) based
on characters.
* io.c (rb_io_getc): getc to return one-character string.
for older behavior, getbyte method added.
* ext/stringio/stringio.c (strio_getc): ditto.
* io.c (rb_io_ungetc): allow pushing arbitrary string at the
current reading point.
* ext/stringio/stringio.c (strio_ungetc): ditto.
* ext/strscan/strscan.c: encoding support.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@13261 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-08-25 07:29:39 +04:00
Copyright ( C ) 1993 - 2007 Yukihiro Matsumoto
2000-05-01 13:42:38 +04:00
Copyright ( C ) 2000 Network Applied Communication Laboratory , Inc .
2000-05-09 08:53:16 +04:00
Copyright ( C ) 2000 Information - technology Promotion Agency , Japan
2000-05-01 13:42:38 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-01-16 15:13:05 +03:00
2020-05-08 12:31:09 +03:00
# include "ruby/internal/config.h"
2019-12-04 11:16:30 +03:00
# include <stddef.h>
2020-05-08 12:31:09 +03:00
# include "ruby/internal/stdbool.h"
2020-04-08 07:28:13 +03:00
# include "ccan/list/list.h"
2010-10-26 21:27:32 +04:00
# include "constant.h"
2019-12-04 11:16:30 +03:00
# include "debug_counter.h"
2013-05-02 11:54:17 +04:00
# include "id.h"
2015-11-03 00:26:26 +03:00
# include "id_table.h"
2019-12-04 11:16:30 +03:00
# include "internal.h"
# include "internal/class.h"
# include "internal/compilers.h"
# include "internal/error.h"
# include "internal/eval.h"
# include "internal/hash.h"
2020-08-17 07:51:23 +03:00
# include "internal/object.h"
2019-12-04 11:16:30 +03:00
# include "internal/re.h"
# include "internal/symbol.h"
# include "internal/thread.h"
# include "internal/variable.h"
# include "ruby/encoding.h"
# include "ruby/st.h"
# include "ruby/util.h"
2022-10-03 18:14:32 +03:00
# include "shape.h"
2023-07-06 14:25:21 +03:00
# include "symbol.h"
fastpath for ivar read of FL_EXIVAR objects.
vm_getivar() provides fastpath for T_OBJECT by caching an index
of ivar. This patch also provides fastpath for FL_EXIVAR objects.
FL_EXIVAR objects have an each ivar array and index can be cached
as T_OBJECT. To access this ivar array, generic_iv_tbl is exposed
by rb_ivar_generic_ivtbl() (declared in variable.h which is newly
introduced).
Benchmark script:
Benchmark.driver(repeat_count: 3){|x|
x.executable name: 'clean', command: %w'../clean/miniruby'
x.executable name: 'trunk', command: %w'./miniruby'
objs = [Object.new, 'str', {a: 1, b: 2}, [1, 2]]
objs.each.with_index{|obj, i|
rep = obj.inspect
rep = 'Object.new' if /\#/ =~ rep
x.prelude str = %Q{
v#{i} = #{rep}
def v#{i}.foo
@iv # ivar access method (attr_reader)
end
v#{i}.instance_variable_set(:@iv, :iv)
}
puts str
x.report %Q{
v#{i}.foo
}
}
}
Result:
v0.foo # T_OBJECT
clean: 85387141.8 i/s
trunk: 85249373.6 i/s - 1.00x slower
v1.foo # T_STRING
trunk: 57894407.5 i/s
clean: 39957178.6 i/s - 1.45x slower
v2.foo # T_HASH
trunk: 56629413.2 i/s
clean: 39227088.9 i/s - 1.44x slower
v3.foo # T_ARRAY
trunk: 55797530.2 i/s
clean: 38263572.9 i/s - 1.46x slower
2019-11-28 21:02:44 +03:00
# include "variable.h"
2019-12-04 11:16:30 +03:00
# include "vm_core.h"
2020-11-17 10:40:47 +03:00
# include "ractor_core.h"
2020-10-14 04:43:13 +03:00
# include "vm_sync.h"
1998-01-16 15:13:05 +03:00
2021-06-01 20:34:06 +03:00
RUBY_EXTERN rb_serial_t ruby_vm_global_cvar_state ;
# define GET_GLOBAL_CVAR_STATE() (ruby_vm_global_cvar_state)
2020-05-07 20:07:57 +03:00
typedef void rb_gvar_compact_t ( void * var ) ;
2018-01-18 12:44:52 +03:00
static struct rb_id_table * rb_global_tbl ;
2023-01-11 17:23:03 +03:00
static ID autoload ;
2022-05-15 07:07:12 +03:00
// This hash table maps file paths to loadable features. We use this to track
// autoload state until it's no longer needed.
2022-05-16 15:50:02 +03:00
// feature (file path) => struct autoload_data
static VALUE autoload_features ;
2022-05-15 07:07:12 +03:00
// This mutex is used to protect autoloading state. We use a global mutex which
// is held until a per-feature mutex can be created. This ensures there are no
// race conditions relating to autoload state.
static VALUE autoload_mutex ;
1998-01-16 15:13:05 +03:00
2015-04-13 10:54:59 +03:00
static void check_before_mod_set ( VALUE , ID , VALUE , const char * ) ;
2015-04-13 10:54:52 +03:00
static void setup_const_entry ( rb_const_entry_t * , VALUE , VALUE , rb_const_flag_t ) ;
2016-06-19 04:54:42 +03:00
static VALUE rb_const_search ( VALUE klass , ID id , int exclude , int recurse , int visibility ) ;
2020-03-09 20:22:11 +03:00
static st_table * generic_iv_tbl_ ;
2015-05-30 02:42:49 +03:00
1998-01-16 15:13:05 +03:00
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
Init_var_tables ( void )
1998-01-16 15:13:05 +03:00
{
2015-11-03 00:26:26 +03:00
rb_global_tbl = rb_id_table_create ( 0 ) ;
2020-03-09 20:22:11 +03:00
generic_iv_tbl_ = st_init_numtable ( ) ;
2014-07-03 07:38:10 +04:00
autoload = rb_intern_const ( " __autoload__ " ) ;
2022-05-15 07:07:12 +03:00
autoload_mutex = rb_mutex_new ( ) ;
rb_obj_hide ( autoload_mutex ) ;
2024-03-03 12:46:46 +03:00
rb_vm_register_global_object ( autoload_mutex ) ;
2022-05-15 07:07:12 +03:00
2022-05-16 15:50:02 +03:00
autoload_features = rb_ident_hash_new ( ) ;
rb_obj_hide ( autoload_features ) ;
2024-03-03 12:46:46 +03:00
rb_vm_register_global_object ( autoload_features ) ;
1998-01-16 15:13:05 +03:00
}
2019-03-09 03:12:52 +03:00
static inline bool
rb_namespace_p ( VALUE obj )
{
if ( RB_SPECIAL_CONST_P ( obj ) ) return false ;
switch ( RB_BUILTIN_TYPE ( obj ) ) {
case T_MODULE : case T_CLASS : return true ;
2020-04-08 09:13:37 +03:00
default : break ;
2019-03-09 03:12:52 +03:00
}
return false ;
}
2012-07-31 13:22:36 +04:00
/**
* Returns + classpath + of _klass_ , if it is named , or + nil + for
2019-04-12 06:46:28 +03:00
* anonymous + class + / + module + . A named + classpath + may contain
* an anonymous component , but the last component is guaranteed
* to not be anonymous . < code > * permanent < / code > is set to 1
* if + classpath + has no anonymous components . There is no builtin
* Ruby level APIs that can change a permanent + classpath + .
2012-07-31 13:22:36 +04:00
*/
1998-01-16 15:13:05 +03:00
static VALUE
2023-01-11 17:23:03 +03:00
classname ( VALUE klass , bool * permanent )
1998-01-16 15:13:05 +03:00
{
2023-01-11 17:23:03 +03:00
* permanent = false ;
2022-11-01 00:05:37 +03:00
2023-01-11 17:23:03 +03:00
VALUE classpath = RCLASS_EXT ( klass ) - > classpath ;
if ( classpath = = 0 ) return Qnil ;
* permanent = RCLASS_EXT ( klass ) - > permanent_classpath ;
2022-11-01 00:05:37 +03:00
2023-01-11 17:23:03 +03:00
return classpath ;
1998-01-16 15:13:05 +03:00
}
2024-02-15 14:04:23 +03:00
VALUE
rb_mod_name0 ( VALUE klass , bool * permanent )
{
return classname ( klass , permanent ) ;
}
2003-12-28 09:33:07 +03:00
/*
* call - seq :
2024-03-09 15:58:01 +03:00
* mod . name - > string or nil
2009-02-15 05:45:31 +03:00
*
2024-03-09 15:58:01 +03:00
* Returns the name of the module < i > mod < / i > . Returns + nil + for anonymous modules .
2003-12-28 09:33:07 +03:00
*/
1998-01-16 15:13:05 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_mod_name ( VALUE mod )
1998-01-16 15:19:22 +03:00
{
2023-01-11 17:23:03 +03:00
bool permanent ;
2019-09-26 11:41:43 +03:00
return classname ( mod , & permanent ) ;
1998-01-16 15:19:22 +03:00
}
2024-03-09 15:58:49 +03:00
// Similar to logic in rb_mod_const_get().
2023-07-06 14:25:21 +03:00
static bool
is_constant_path ( VALUE name )
{
const char * path = RSTRING_PTR ( name ) ;
const char * pend = RSTRING_END ( name ) ;
rb_encoding * enc = rb_enc_get ( name ) ;
const char * p = path ;
if ( p > = pend | | ! * p ) {
return false ;
}
while ( p < pend ) {
if ( p + 2 < = pend & & p [ 0 ] = = ' : ' & & p [ 1 ] = = ' : ' ) {
p + = 2 ;
}
const char * pbeg = p ;
while ( p < pend & & * p ! = ' : ' ) p + + ;
if ( pbeg = = p ) return false ;
if ( rb_enc_symname_type ( pbeg , p - pbeg , enc , 0 ) ! = ID_CONST ) {
return false ;
}
}
return true ;
}
2023-06-21 10:49:51 +03:00
/*
* call - seq :
* mod . set_temporary_name ( string ) - > self
* mod . set_temporary_name ( nil ) - > self
*
2023-12-15 00:01:48 +03:00
* Sets the temporary name of the module . This name is reflected in
* introspection of the module and the values that are related to it , such
* as instances , constants , and methods .
2023-06-21 10:49:51 +03:00
*
2024-03-09 15:58:49 +03:00
* The name should be + nil + or a non - empty string that is not a valid constant
* path ( to avoid confusing between permanent and temporary names ) .
2023-06-21 10:49:51 +03:00
*
2023-12-15 00:01:48 +03:00
* The method can be useful to distinguish dynamically generated classes and
* modules without assigning them to constants .
2023-06-21 10:49:51 +03:00
*
2023-12-15 00:01:48 +03:00
* If the module is given a permanent name by assigning it to a constant ,
* the temporary name is discarded . A temporary name can ' t be assigned to
* modules that have a permanent name .
2023-06-21 10:49:51 +03:00
*
2023-12-15 00:01:48 +03:00
* If the given name is + nil + , the module becomes anonymous again .
2023-06-21 10:49:51 +03:00
*
* Example :
*
* m = Module . new # = > # < Module : 0x0000000102c68f38 >
* m . name # = > nil
*
* m . set_temporary_name ( " fake_name " ) # = > fake_name
* m . name # = > " fake_name "
*
* m . set_temporary_name ( nil ) # = > # < Module : 0x0000000102c68f38 >
* m . name # = > nil
*
2023-12-15 00:01:48 +03:00
* c = Class . new
* c . set_temporary_name ( " MyClass(with description) " )
2023-06-21 10:49:51 +03:00
*
2023-12-15 00:01:48 +03:00
* c . new # = > # < MyClass ( with description ) : 0x0 . . . . >
2023-06-21 10:49:51 +03:00
*
2023-12-15 00:01:48 +03:00
* c : : M = m
* c : : M . name # = > " MyClass(with description)::M "
*
* # Assigning to a constant replaces the name with a permanent one
* C = c
*
* C . name # = > " C "
* C : : M . name # = > " C::M "
* c . new # = > # < C : 0x0 . . . . >
2023-06-21 10:49:51 +03:00
*/
VALUE
rb_mod_set_temporary_name ( VALUE mod , VALUE name )
{
// We don't allow setting the name if the classpath is already permanent:
if ( RCLASS_EXT ( mod ) - > permanent_classpath ) {
rb_raise ( rb_eRuntimeError , " can't change permanent name " ) ;
}
if ( NIL_P ( name ) ) {
// Set the temporary classpath to NULL (anonymous):
RCLASS_SET_CLASSPATH ( mod , 0 , FALSE ) ;
2024-01-07 18:50:41 +03:00
}
else {
2023-06-21 10:49:51 +03:00
// Ensure the name is a string:
StringValue ( name ) ;
if ( RSTRING_LEN ( name ) = = 0 ) {
rb_raise ( rb_eArgError , " empty class/module name " ) ;
}
2023-07-06 14:25:21 +03:00
if ( is_constant_path ( name ) ) {
2023-07-06 17:34:28 +03:00
rb_raise ( rb_eArgError , " the temporary name must not be a constant path to avoid confusion " ) ;
2023-06-21 10:49:51 +03:00
}
// Set the temporary classpath to the given name:
RCLASS_SET_CLASSPATH ( mod , name , FALSE ) ;
}
return mod ;
}
2015-02-28 09:42:29 +03:00
static VALUE
make_temporary_path ( VALUE obj , VALUE klass )
{
VALUE path ;
switch ( klass ) {
case Qnil :
path = rb_sprintf ( " #<Class:%p> " , ( void * ) obj ) ;
break ;
case Qfalse :
path = rb_sprintf ( " #<Module:%p> " , ( void * ) obj ) ;
break ;
default :
path = rb_sprintf ( " #<% " PRIsVALUE " :%p> " , klass , ( void * ) obj ) ;
break ;
}
OBJ_FREEZE ( path ) ;
return path ;
}
2019-04-12 06:46:28 +03:00
typedef VALUE ( * fallback_func ) ( VALUE obj , VALUE name ) ;
2012-12-01 06:13:06 +04:00
2012-07-30 09:24:24 +04:00
static VALUE
2023-01-11 17:23:03 +03:00
rb_tmp_class_path ( VALUE klass , bool * permanent , fallback_func fallback )
1998-01-16 15:13:05 +03:00
{
2012-07-31 13:22:36 +04:00
VALUE path = classname ( klass , permanent ) ;
1998-01-16 15:13:05 +03:00
2012-07-30 09:24:24 +04:00
if ( ! NIL_P ( path ) ) {
return path ;
}
2024-03-09 15:59:24 +03:00
if ( RB_TYPE_P ( klass , T_MODULE ) ) {
if ( rb_obj_class ( klass ) = = rb_cModule ) {
path = Qfalse ;
}
else {
bool perm ;
path = rb_tmp_class_path ( RBASIC ( klass ) - > klass , & perm , fallback ) ;
2001-10-03 11:19:19 +04:00
}
1998-01-16 15:13:05 +03:00
}
2024-03-09 15:59:24 +03:00
* permanent = false ;
return fallback ( klass , path ) ;
1998-01-16 15:13:05 +03:00
}
2012-07-30 09:24:24 +04:00
VALUE
rb_class_path ( VALUE klass )
{
2023-01-11 17:23:03 +03:00
bool permanent ;
2019-04-12 06:46:28 +03:00
VALUE path = rb_tmp_class_path ( klass , & permanent , make_temporary_path ) ;
2012-07-31 13:22:36 +04:00
if ( ! NIL_P ( path ) ) path = rb_str_dup ( path ) ;
return path ;
2012-07-30 09:24:24 +04:00
}
2013-12-09 07:25:38 +04:00
VALUE
rb_class_path_cached ( VALUE klass )
{
2021-06-01 10:34:56 +03:00
return rb_mod_name ( klass ) ;
2013-12-09 07:25:38 +04:00
}
2015-02-28 09:42:29 +03:00
static VALUE
2019-04-12 06:46:28 +03:00
no_fallback ( VALUE obj , VALUE name )
2015-02-28 09:42:29 +03:00
{
return name ;
}
VALUE
rb_search_class_path ( VALUE klass )
{
2023-01-11 17:23:03 +03:00
bool permanent ;
2019-04-12 06:46:28 +03:00
return rb_tmp_class_path ( klass , & permanent , no_fallback ) ;
}
2019-04-12 06:46:28 +03:00
static VALUE
build_const_pathname ( VALUE head , VALUE tail )
{
VALUE path = rb_str_dup ( head ) ;
rb_str_cat2 ( path , " :: " ) ;
rb_str_append ( path , tail ) ;
2021-09-23 23:17:56 +03:00
return rb_fstring ( path ) ;
2019-04-12 06:46:28 +03:00
}
static VALUE
build_const_path ( VALUE head , ID tail )
{
return build_const_pathname ( head , rb_id2str ( tail ) ) ;
}
2009-07-30 11:45:42 +04:00
void
rb_set_class_path_string ( VALUE klass , VALUE under , VALUE name )
{
2023-01-11 17:23:03 +03:00
bool permanent = true ;
2009-07-30 11:45:42 +04:00
2023-01-11 17:23:03 +03:00
VALUE str ;
2009-07-30 11:45:42 +04:00
if ( under = = rb_cObject ) {
str = rb_str_new_frozen ( name ) ;
}
else {
2019-08-11 00:32:49 +03:00
str = rb_tmp_class_path ( under , & permanent , make_temporary_path ) ;
2019-04-12 06:46:28 +03:00
str = build_const_pathname ( str , name ) ;
2009-07-30 11:45:42 +04:00
}
2023-01-11 17:23:03 +03:00
RCLASS_SET_CLASSPATH ( klass , str , permanent ) ;
2009-07-30 11:45:42 +04:00
}
1998-01-16 15:13:05 +03:00
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_set_class_path ( VALUE klass , VALUE under , const char * name )
1998-01-16 15:13:05 +03:00
{
2019-08-11 00:32:49 +03:00
VALUE str = rb_str_new2 ( name ) ;
2007-06-07 12:21:01 +04:00
OBJ_FREEZE ( str ) ;
2019-08-11 00:32:49 +03:00
rb_set_class_path_string ( klass , under , str ) ;
1998-01-16 15:13:05 +03:00
}
VALUE
2009-08-10 01:55:55 +04:00
rb_path_to_class ( VALUE pathname )
1998-01-16 15:13:05 +03:00
{
2009-08-10 01:55:55 +04:00
rb_encoding * enc = rb_enc_get ( pathname ) ;
2016-06-19 05:19:45 +03:00
const char * pbeg , * pend , * p , * path = RSTRING_PTR ( pathname ) ;
2002-09-05 13:42:56 +04:00
ID id ;
VALUE c = rb_cObject ;
1999-11-29 09:33:02 +03:00
2009-08-10 01:55:55 +04:00
if ( ! rb_enc_asciicompat ( enc ) ) {
rb_raise ( rb_eArgError , " invalid class path encoding (non ASCII) " ) ;
}
pbeg = p = path ;
2016-06-19 05:19:45 +03:00
pend = path + RSTRING_LEN ( pathname ) ;
if ( path = = pend | | path [ 0 ] = = ' # ' ) {
2012-12-22 19:04:57 +04:00
rb_raise ( rb_eArgError , " can't retrieve anonymous class % " PRIsVALUE ,
QUOTE ( pathname ) ) ;
1998-01-16 15:13:05 +03:00
}
2016-06-19 05:19:45 +03:00
while ( p < pend ) {
while ( p < pend & & * p ! = ' : ' ) p + + ;
2014-07-09 10:14:41 +04:00
id = rb_check_id_cstr ( pbeg , p - pbeg , enc ) ;
2016-06-19 05:19:45 +03:00
if ( p < pend & & p [ 0 ] = = ' : ' ) {
if ( ( size_t ) ( pend - p ) < 2 | | p [ 1 ] ! = ' : ' ) goto undefined_class ;
2002-09-05 13:42:56 +04:00
p + = 2 ;
pbeg = p ;
}
2016-06-19 04:54:42 +03:00
if ( ! id ) {
2020-06-19 09:51:17 +03:00
goto undefined_class ;
2002-09-05 13:42:56 +04:00
}
2016-06-19 04:54:42 +03:00
c = rb_const_search ( c , id , TRUE , FALSE , FALSE ) ;
2022-11-15 07:24:08 +03:00
if ( UNDEF_P ( c ) ) goto undefined_class ;
2019-03-09 03:12:53 +03:00
if ( ! rb_namespace_p ( c ) ) {
2012-12-22 19:04:57 +04:00
rb_raise ( rb_eTypeError , " % " PRIsVALUE " does not refer to class/module " ,
pathname ) ;
2002-09-05 13:42:56 +04:00
}
1999-11-29 09:33:02 +03:00
}
2012-12-04 11:23:20 +04:00
RB_GC_GUARD ( pathname ) ;
2002-09-05 13:42:56 +04:00
1999-11-29 09:33:02 +03:00
return c ;
2020-06-19 09:51:17 +03:00
undefined_class :
rb_raise ( rb_eArgError , " undefined class/module % " PRIsVALUE ,
rb_str_subseq ( pathname , 0 , p - path ) ) ;
2020-06-24 10:23:59 +03:00
UNREACHABLE_RETURN ( Qundef ) ;
1998-01-16 15:13:05 +03:00
}
2009-08-10 01:55:55 +04:00
VALUE
rb_path2class ( const char * path )
{
2009-09-09 08:09:09 +04:00
return rb_path_to_class ( rb_str_new_cstr ( path ) ) ;
2009-08-10 01:55:55 +04:00
}
2004-01-19 12:19:31 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_class_name ( VALUE klass )
2004-01-19 12:19:31 +03:00
{
return rb_class_path ( rb_class_real ( klass ) ) ;
}
2008-05-31 13:28:20 +04:00
const char *
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_class2name ( VALUE klass )
1998-01-16 15:13:05 +03:00
{
2023-01-11 17:23:03 +03:00
bool permanent ;
2019-04-12 06:46:28 +03:00
VALUE path = rb_tmp_class_path ( rb_class_real ( klass ) , & permanent , make_temporary_path ) ;
2013-10-19 23:21:05 +04:00
if ( NIL_P ( path ) ) return NULL ;
return RSTRING_PTR ( path ) ;
1998-01-16 15:13:05 +03:00
}
2008-05-31 13:28:20 +04:00
const char *
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_obj_classname ( VALUE obj )
2003-01-31 07:00:17 +03:00
{
2003-01-31 07:16:51 +03:00
return rb_class2name ( CLASS_OF ( obj ) ) ;
2003-01-31 07:00:17 +03:00
}
1998-01-16 15:13:05 +03:00
struct trace_var {
1998-01-16 15:19:22 +03:00
int removed ;
2008-09-26 05:35:57 +04:00
void ( * func ) ( VALUE arg , VALUE val ) ;
2001-03-19 06:20:24 +03:00
VALUE data ;
1998-01-16 15:13:05 +03:00
struct trace_var * next ;
} ;
2015-11-30 02:53:43 +03:00
struct rb_global_variable {
int counter ;
2014-03-09 01:44:24 +04:00
int block_trace ;
2019-08-27 05:16:52 +03:00
VALUE * data ;
2015-11-30 02:53:43 +03:00
rb_gvar_getter_t * getter ;
rb_gvar_setter_t * setter ;
rb_gvar_marker_t * marker ;
2020-05-07 20:07:57 +03:00
rb_gvar_compact_t * compactor ;
1998-01-16 15:13:05 +03:00
struct trace_var * trace ;
} ;
2020-07-03 10:56:44 +03:00
struct rb_global_entry {
struct rb_global_variable * var ;
ID id ;
2020-03-09 20:22:11 +03:00
bool ractor_local ;
2020-07-03 10:56:44 +03:00
} ;
2023-10-12 21:15:53 +03:00
static enum rb_id_table_iterator_result
2024-06-10 17:02:05 +03:00
free_global_entry_i ( VALUE val , void * arg )
2023-10-12 21:15:53 +03:00
{
struct rb_global_entry * entry = ( struct rb_global_entry * ) val ;
if ( entry - > var - > counter = = 1 ) {
ruby_xfree ( entry - > var ) ;
}
else {
entry - > var - > counter - - ;
}
ruby_xfree ( entry ) ;
return ID_TABLE_DELETE ;
}
void
rb_free_rb_global_tbl ( void )
{
2024-06-10 17:02:05 +03:00
rb_id_table_foreach_values ( rb_global_tbl , free_global_entry_i , 0 ) ;
2023-10-12 21:15:53 +03:00
rb_id_table_free ( rb_global_tbl ) ;
}
void
rb_free_generic_iv_tbl_ ( void )
{
st_free_table ( generic_iv_tbl_ ) ;
}
2018-11-25 07:05:09 +03:00
static struct rb_global_entry *
rb_find_global_entry ( ID id )
1998-01-16 15:13:05 +03:00
{
2015-11-30 02:53:43 +03:00
struct rb_global_entry * entry ;
2015-11-03 00:26:26 +03:00
VALUE data ;
1998-01-16 15:13:05 +03:00
2020-03-09 20:22:11 +03:00
if ( ! rb_id_table_lookup ( rb_global_tbl , id , & data ) ) {
entry = NULL ;
}
else {
entry = ( struct rb_global_entry * ) data ;
RUBY_ASSERT ( entry ! = NULL ) ;
}
if ( UNLIKELY ( ! rb_ractor_main_p ( ) ) & & ( ! entry | | ! entry - > ractor_local ) ) {
2020-12-21 12:06:28 +03:00
rb_raise ( rb_eRactorIsolationError , " can not access global variables %s from non-main Ractors " , rb_id2name ( id ) ) ;
2018-11-25 07:05:09 +03:00
}
2020-03-09 20:22:11 +03:00
2018-11-25 07:05:09 +03:00
return entry ;
}
2020-03-09 20:22:11 +03:00
void
rb_gvar_ractor_local ( const char * name )
{
struct rb_global_entry * entry = rb_find_global_entry ( rb_intern ( name ) ) ;
entry - > ractor_local = true ;
}
2020-05-07 20:07:57 +03:00
static void
rb_gvar_undef_compactor ( void * var )
{
}
2020-07-03 10:56:44 +03:00
static struct rb_global_entry *
2018-11-25 07:05:09 +03:00
rb_global_entry ( ID id )
{
struct rb_global_entry * entry = rb_find_global_entry ( id ) ;
if ( ! entry ) {
2015-11-30 02:53:43 +03:00
struct rb_global_variable * var ;
entry = ALLOC ( struct rb_global_entry ) ;
var = ALLOC ( struct rb_global_variable ) ;
1998-01-16 15:13:05 +03:00
entry - > id = id ;
2001-10-22 20:20:14 +04:00
entry - > var = var ;
2020-03-09 20:22:11 +03:00
entry - > ractor_local = false ;
2001-10-22 20:20:14 +04:00
var - > counter = 1 ;
var - > data = 0 ;
2015-11-30 02:53:43 +03:00
var - > getter = rb_gvar_undef_getter ;
var - > setter = rb_gvar_undef_setter ;
var - > marker = rb_gvar_undef_marker ;
2020-05-07 20:07:57 +03:00
var - > compactor = rb_gvar_undef_compactor ;
1998-01-16 15:13:05 +03:00
2001-10-22 20:20:14 +04:00
var - > block_trace = 0 ;
var - > trace = 0 ;
2020-03-09 20:22:11 +03:00
rb_id_table_insert ( rb_global_tbl , id , ( VALUE ) entry ) ;
1998-01-16 15:13:05 +03:00
}
return entry ;
}
2008-10-14 15:45:32 +04:00
VALUE
2019-08-27 05:16:52 +03:00
rb_gvar_undef_getter ( ID id , VALUE * _ )
1998-01-16 15:13:05 +03:00
{
2024-01-19 10:03:38 +03:00
rb_warning ( " global variable '% " PRIsVALUE " ' not initialized " , QUOTE_ID ( id ) ) ;
2002-04-11 14:03:01 +04:00
1998-01-16 15:13:05 +03:00
return Qnil ;
}
2020-05-07 20:07:57 +03:00
static void
rb_gvar_val_compactor ( void * _var )
{
struct rb_global_variable * var = ( struct rb_global_variable * ) _var ;
VALUE obj = ( VALUE ) var - > data ;
if ( obj ) {
VALUE new = rb_gc_location ( obj ) ;
if ( new ! = obj ) {
var - > data = ( void * ) new ;
}
}
}
2008-10-14 15:45:32 +04:00
void
2019-08-27 05:16:52 +03:00
rb_gvar_undef_setter ( VALUE val , ID id , VALUE * _ )
1998-01-16 15:13:05 +03:00
{
2019-08-27 05:16:52 +03:00
struct rb_global_variable * var = rb_global_entry ( id ) - > var ;
2015-11-30 02:53:43 +03:00
var - > getter = rb_gvar_val_getter ;
var - > setter = rb_gvar_val_setter ;
var - > marker = rb_gvar_val_marker ;
2020-05-07 20:07:57 +03:00
var - > compactor = rb_gvar_val_compactor ;
1998-01-16 15:13:05 +03:00
2001-10-22 20:20:14 +04:00
var - > data = ( void * ) val ;
1998-01-16 15:13:05 +03:00
}
2008-10-14 15:45:32 +04:00
void
2015-11-30 02:53:43 +03:00
rb_gvar_undef_marker ( VALUE * var )
1998-01-16 15:13:05 +03:00
{
}
2008-10-14 15:45:32 +04:00
VALUE
2019-08-27 05:16:52 +03:00
rb_gvar_val_getter ( ID id , VALUE * data )
1998-01-16 15:13:05 +03:00
{
2008-09-26 05:35:57 +04:00
return ( VALUE ) data ;
1998-01-16 15:13:05 +03:00
}
2008-10-14 15:45:32 +04:00
void
2019-08-27 05:16:52 +03:00
rb_gvar_val_setter ( VALUE val , ID id , VALUE * _ )
1998-01-16 15:13:05 +03:00
{
2019-08-27 05:16:52 +03:00
struct rb_global_variable * var = rb_global_entry ( id ) - > var ;
2001-10-22 20:20:14 +04:00
var - > data = ( void * ) val ;
1998-01-16 15:13:05 +03:00
}
2008-10-14 15:45:32 +04:00
void
2015-11-30 02:53:43 +03:00
rb_gvar_val_marker ( VALUE * var )
1998-01-16 15:13:05 +03:00
{
2008-09-26 05:35:57 +04:00
VALUE data = ( VALUE ) var ;
2020-05-07 20:07:57 +03:00
if ( data ) rb_gc_mark_movable ( data ) ;
1998-01-16 15:13:05 +03:00
}
2008-10-14 15:45:32 +04:00
VALUE
2019-08-27 05:16:52 +03:00
rb_gvar_var_getter ( ID id , VALUE * var )
1998-01-16 15:13:05 +03:00
{
2001-11-13 11:19:52 +03:00
if ( ! var ) return Qnil ;
1998-01-16 15:13:05 +03:00
return * var ;
}
2008-10-14 15:45:32 +04:00
void
2019-08-27 05:16:52 +03:00
rb_gvar_var_setter ( VALUE val , ID id , VALUE * data )
1998-01-16 15:13:05 +03:00
{
2019-08-27 05:16:52 +03:00
* data = val ;
1998-01-16 15:13:05 +03:00
}
2008-10-14 15:45:32 +04:00
void
2015-11-30 02:53:43 +03:00
rb_gvar_var_marker ( VALUE * var )
1998-01-16 15:13:05 +03:00
{
1999-01-20 07:59:39 +03:00
if ( var ) rb_gc_mark_maybe ( * var ) ;
1998-01-16 15:13:05 +03:00
}
2008-10-14 15:45:32 +04:00
void
2019-08-27 05:16:52 +03:00
rb_gvar_readonly_setter ( VALUE v , ID id , VALUE * _ )
1998-01-16 15:13:05 +03:00
{
2012-12-22 19:04:57 +04:00
rb_name_error ( id , " % " PRIsVALUE " is a read-only variable " , QUOTE_ID ( id ) ) ;
1998-01-16 15:13:05 +03:00
}
2015-11-03 00:26:26 +03:00
static enum rb_id_table_iterator_result
mark_global_entry ( VALUE v , void * ignored )
1998-01-16 15:13:05 +03:00
{
2015-11-30 02:53:43 +03:00
struct rb_global_entry * entry = ( struct rb_global_entry * ) v ;
1998-01-16 15:13:05 +03:00
struct trace_var * trace ;
2015-11-30 02:53:43 +03:00
struct rb_global_variable * var = entry - > var ;
1998-01-16 15:13:05 +03:00
2001-10-22 20:20:14 +04:00
( * var - > marker ) ( var - > data ) ;
trace = var - > trace ;
1998-01-16 15:13:05 +03:00
while ( trace ) {
1999-01-20 07:59:39 +03:00
if ( trace - > data ) rb_gc_mark_maybe ( trace - > data ) ;
1998-01-16 15:13:05 +03:00
trace = trace - > next ;
}
2015-11-03 00:26:26 +03:00
return ID_TABLE_CONTINUE ;
1998-01-16 15:13:05 +03:00
}
2023-05-15 11:22:01 +03:00
# define gc_mark_table(task) \
if ( rb_global_tbl ) { rb_id_table_foreach_values ( rb_global_tbl , task # # _global_entry , 0 ) ; }
1998-01-16 15:13:05 +03:00
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_gc_mark_global_tbl ( void )
1998-01-16 15:13:05 +03:00
{
2023-05-15 11:22:01 +03:00
gc_mark_table ( mark ) ;
1998-01-16 15:13:05 +03:00
}
2020-05-07 20:07:57 +03:00
static enum rb_id_table_iterator_result
update_global_entry ( VALUE v , void * ignored )
{
struct rb_global_entry * entry = ( struct rb_global_entry * ) v ;
struct rb_global_variable * var = entry - > var ;
( * var - > compactor ) ( var ) ;
return ID_TABLE_CONTINUE ;
}
void
rb_gc_update_global_tbl ( void )
{
2023-05-15 11:22:01 +03:00
gc_mark_table ( update ) ;
2020-05-07 20:07:57 +03:00
}
1998-01-16 15:13:05 +03:00
static ID
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
global_id ( const char * name )
1998-01-16 15:13:05 +03:00
{
ID id ;
if ( name [ 0 ] = = ' $ ' ) id = rb_intern ( name ) ;
else {
* dir.c, dln.c, parse.y, re.c, ruby.c, sprintf.c, strftime.c,
string.c, util.c, variable.c: use strlcpy, memcpy and snprintf
instead of strcpy, strncpy and sprintf.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@22984 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2009-03-17 04:29:17 +03:00
size_t len = strlen ( name ) ;
2018-11-25 06:36:10 +03:00
VALUE vbuf = 0 ;
char * buf = ALLOCV_N ( char , vbuf , len + 1 ) ;
1998-01-16 15:13:05 +03:00
buf [ 0 ] = ' $ ' ;
* dir.c, dln.c, parse.y, re.c, ruby.c, sprintf.c, strftime.c,
string.c, util.c, variable.c: use strlcpy, memcpy and snprintf
instead of strcpy, strncpy and sprintf.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@22984 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2009-03-17 04:29:17 +03:00
memcpy ( buf + 1 , name , len ) ;
id = rb_intern2 ( buf , len + 1 ) ;
2018-11-25 06:36:10 +03:00
ALLOCV_END ( vbuf ) ;
1998-01-16 15:13:05 +03:00
}
return id ;
}
2018-11-25 07:05:09 +03:00
static ID
find_global_id ( const char * name )
{
ID id ;
size_t len = strlen ( name ) ;
if ( name [ 0 ] = = ' $ ' ) {
id = rb_check_id_cstr ( name , len , NULL ) ;
}
else {
VALUE vbuf = 0 ;
char * buf = ALLOCV_N ( char , vbuf , len + 1 ) ;
buf [ 0 ] = ' $ ' ;
memcpy ( buf + 1 , name , len ) ;
id = rb_check_id_cstr ( buf , len + 1 , NULL ) ;
ALLOCV_END ( vbuf ) ;
}
return id ;
}
1998-01-16 15:13:05 +03:00
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_define_hooked_variable (
const char * name ,
VALUE * var ,
2019-08-27 05:16:52 +03:00
rb_gvar_getter_t * getter ,
rb_gvar_setter_t * setter )
1998-01-16 15:13:05 +03:00
{
2009-02-15 05:45:31 +03:00
volatile VALUE tmp = var ? * var : Qnil ;
ID id = global_id ( name ) ;
2015-11-30 02:53:43 +03:00
struct rb_global_variable * gvar = rb_global_entry ( id ) - > var ;
1998-01-16 15:13:05 +03:00
2001-10-22 20:20:14 +04:00
gvar - > data = ( void * ) var ;
2015-11-30 02:53:43 +03:00
gvar - > getter = getter ? ( rb_gvar_getter_t * ) getter : rb_gvar_var_getter ;
gvar - > setter = setter ? ( rb_gvar_setter_t * ) setter : rb_gvar_var_setter ;
gvar - > marker = rb_gvar_var_marker ;
2008-05-02 18:57:23 +04:00
RB_GC_GUARD ( tmp ) ;
1998-01-16 15:13:05 +03:00
}
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_define_variable ( const char * name , VALUE * var )
1998-01-16 15:13:05 +03:00
{
rb_define_hooked_variable ( name , var , 0 , 0 ) ;
}
void
2014-06-18 10:16:39 +04:00
rb_define_readonly_variable ( const char * name , const VALUE * var )
1998-01-16 15:13:05 +03:00
{
2015-11-30 02:53:43 +03:00
rb_define_hooked_variable ( name , ( VALUE * ) var , 0 , rb_gvar_readonly_setter ) ;
1998-01-16 15:13:05 +03:00
}
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_define_virtual_variable (
const char * name ,
2019-08-27 05:16:52 +03:00
rb_gvar_getter_t * getter ,
rb_gvar_setter_t * setter )
1998-01-16 15:13:05 +03:00
{
2015-11-30 02:53:43 +03:00
if ( ! getter ) getter = rb_gvar_val_getter ;
if ( ! setter ) setter = rb_gvar_readonly_setter ;
1998-01-16 15:13:05 +03:00
rb_define_hooked_variable ( name , 0 , getter , setter ) ;
}
static void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_trace_eval ( VALUE cmd , VALUE val )
1998-01-16 15:13:05 +03:00
{
2019-09-26 01:49:21 +03:00
rb_eval_cmd_kw ( cmd , rb_ary_new3 ( 1 , val ) , RB_NO_KEYWORDS ) ;
1998-01-16 15:13:05 +03:00
}
VALUE
2014-06-18 10:16:39 +04:00
rb_f_trace_var ( int argc , const VALUE * argv )
1998-01-16 15:13:05 +03:00
{
VALUE var , cmd ;
2015-11-30 02:53:43 +03:00
struct rb_global_entry * entry ;
1998-01-16 15:13:05 +03:00
struct trace_var * trace ;
if ( rb_scan_args ( argc , argv , " 11 " , & var , & cmd ) = = 1 ) {
2003-06-16 11:14:50 +04:00
cmd = rb_block_proc ( ) ;
1998-01-16 15:13:05 +03:00
}
if ( NIL_P ( cmd ) ) {
1999-01-20 07:59:39 +03:00
return rb_f_untrace_var ( argc , argv ) ;
1998-01-16 15:13:05 +03:00
}
2001-11-13 11:19:52 +03:00
entry = rb_global_entry ( rb_to_id ( var ) ) ;
1998-01-16 15:13:05 +03:00
trace = ALLOC ( struct trace_var ) ;
2001-10-22 20:20:14 +04:00
trace - > next = entry - > var - > trace ;
1998-01-16 15:13:05 +03:00
trace - > func = rb_trace_eval ;
2001-03-19 06:20:24 +03:00
trace - > data = cmd ;
1998-01-16 15:19:22 +03:00
trace - > removed = 0 ;
2001-10-22 20:20:14 +04:00
entry - > var - > trace = trace ;
1998-01-16 15:13:05 +03:00
return Qnil ;
}
1998-01-16 15:19:22 +03:00
static void
2015-11-30 02:53:43 +03:00
remove_trace ( struct rb_global_variable * var )
1998-01-16 15:19:22 +03:00
{
2001-10-22 20:20:14 +04:00
struct trace_var * trace = var - > trace ;
1998-01-16 15:19:22 +03:00
struct trace_var t ;
struct trace_var * next ;
t . next = trace ;
trace = & t ;
while ( trace - > next ) {
next = trace - > next ;
if ( next - > removed ) {
trace - > next = next - > next ;
* array.c, bignum.c, cont.c, dir.c, dln.c, encoding.c, enumerator.c,
enumerator.c (enumerator_allocate), eval_jump.c, file.c, hash.c,
io.c, load.c, pack.c, proc.c, random.c, re.c, ruby.c, st.c,
string.c, thread.c, thread_pthread.c, time.c, util.c, variable.c,
vm.c, gc.c:
allocated memory objects by xmalloc (ruby_xmalloc) should be
freed by xfree (ruby_xfree).
* ext/curses/curses.c, ext/dbm/dbm.c, ext/digest/digest.c,
ext/gdbm/gdbm.c, ext/json/ext/parser/parser.c,
ext/json/ext/parser/unicode.c, ext/openssl/ossl_cipher.c,
ext/openssl/ossl_hmac.c, ext/openssl/ossl_pkey_ec.c,
ext/sdbm/init.c, ext/strscan/strscan.c, ext/zlib/zlib.c:
ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@17017 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2008-06-08 14:01:40 +04:00
xfree ( next ) ;
1998-01-16 15:19:22 +03:00
}
2001-10-21 17:22:54 +04:00
else {
trace = next ;
}
1998-01-16 15:19:22 +03:00
}
2001-10-22 20:20:14 +04:00
var - > trace = t . next ;
1998-01-16 15:19:22 +03:00
}
1998-01-16 15:13:05 +03:00
VALUE
2014-06-18 10:16:39 +04:00
rb_f_untrace_var ( int argc , const VALUE * argv )
1998-01-16 15:13:05 +03:00
{
VALUE var , cmd ;
ID id ;
2015-11-30 02:53:43 +03:00
struct rb_global_entry * entry ;
1998-01-16 15:13:05 +03:00
struct trace_var * trace ;
rb_scan_args ( argc , argv , " 11 " , & var , & cmd ) ;
2011-07-26 20:05:35 +04:00
id = rb_check_id ( & var ) ;
if ( ! id ) {
2012-12-22 19:04:57 +04:00
rb_name_error_str ( var , " undefined global variable % " PRIsVALUE " " , QUOTE ( var ) ) ;
2011-07-26 20:05:35 +04:00
}
2020-03-09 20:22:11 +03:00
if ( ( entry = rb_find_global_entry ( id ) ) = = NULL ) {
2012-12-22 19:04:57 +04:00
rb_name_error ( id , " undefined global variable % " PRIsVALUE " " , QUOTE_ID ( id ) ) ;
1998-01-16 15:13:05 +03:00
}
1999-01-20 07:59:39 +03:00
2020-03-09 20:22:11 +03:00
trace = entry - > var - > trace ;
1998-01-16 15:13:05 +03:00
if ( NIL_P ( cmd ) ) {
1999-01-20 07:59:39 +03:00
VALUE ary = rb_ary_new ( ) ;
1998-01-16 15:13:05 +03:00
while ( trace ) {
struct trace_var * next = trace - > next ;
1999-01-20 07:59:39 +03:00
rb_ary_push ( ary , ( VALUE ) trace - > data ) ;
1998-01-16 15:19:22 +03:00
trace - > removed = 1 ;
1998-01-16 15:13:05 +03:00
trace = next ;
}
2001-10-22 20:20:14 +04:00
if ( ! entry - > var - > block_trace ) remove_trace ( entry - > var ) ;
1998-01-16 15:13:05 +03:00
return ary ;
}
else {
1998-01-16 15:19:22 +03:00
while ( trace ) {
2001-03-19 06:20:24 +03:00
if ( trace - > data = = cmd ) {
1998-01-16 15:19:22 +03:00
trace - > removed = 1 ;
2001-10-22 20:20:14 +04:00
if ( ! entry - > var - > block_trace ) remove_trace ( entry - > var ) ;
1999-01-20 07:59:39 +03:00
return rb_ary_new3 ( 1 , cmd ) ;
1998-01-16 15:13:05 +03:00
}
1998-01-16 15:19:22 +03:00
trace = trace - > next ;
1998-01-16 15:13:05 +03:00
}
}
return Qnil ;
}
2001-10-17 09:28:02 +04:00
1998-01-16 15:13:05 +03:00
struct trace_data {
struct trace_var * trace ;
VALUE val ;
} ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
1999-01-20 07:59:39 +03:00
static VALUE
2019-08-26 09:20:15 +03:00
trace_ev ( VALUE v )
1998-01-16 15:13:05 +03:00
{
2019-08-26 09:20:15 +03:00
struct trace_data * data = ( void * ) v ;
1998-01-16 15:13:05 +03:00
struct trace_var * trace = data - > trace ;
while ( trace ) {
( * trace - > func ) ( trace - > data , data - > val ) ;
trace = trace - > next ;
}
2012-04-14 04:36:26 +04:00
return Qnil ;
1998-01-16 15:13:05 +03:00
}
1999-01-20 07:59:39 +03:00
static VALUE
2019-08-26 09:20:15 +03:00
trace_en ( VALUE v )
1998-01-16 15:13:05 +03:00
{
2019-08-26 09:20:15 +03:00
struct rb_global_variable * var = ( void * ) v ;
2001-10-22 20:20:14 +04:00
var - > block_trace = 0 ;
remove_trace ( var ) ;
1999-01-20 07:59:39 +03:00
return Qnil ; /* not reached */
1998-01-16 15:13:05 +03:00
}
2020-07-03 10:56:44 +03:00
static VALUE
rb_gvar_set_entry ( struct rb_global_entry * entry , VALUE val )
1998-01-16 15:13:05 +03:00
{
struct trace_data trace ;
2015-11-30 02:53:43 +03:00
struct rb_global_variable * var = entry - > var ;
1998-01-16 15:13:05 +03:00
2019-08-27 05:16:52 +03:00
( * var - > setter ) ( val , entry - > id , var - > data ) ;
1998-01-16 15:13:05 +03:00
2001-10-22 20:20:14 +04:00
if ( var - > trace & & ! var - > block_trace ) {
var - > block_trace = 1 ;
trace . trace = var - > trace ;
1998-01-16 15:13:05 +03:00
trace . val = val ;
2001-10-22 20:20:14 +04:00
rb_ensure ( trace_ev , ( VALUE ) & trace , trace_en , ( VALUE ) var ) ;
1998-01-16 15:13:05 +03:00
}
return val ;
}
VALUE
2020-07-03 10:56:44 +03:00
rb_gvar_set ( ID id , VALUE val )
1998-01-16 15:13:05 +03:00
{
2015-11-30 02:53:43 +03:00
struct rb_global_entry * entry ;
2020-07-03 10:56:44 +03:00
entry = rb_global_entry ( id ) ;
return rb_gvar_set_entry ( entry , val ) ;
}
1998-01-16 15:13:05 +03:00
2020-07-03 10:56:44 +03:00
VALUE
rb_gv_set ( const char * name , VALUE val )
{
return rb_gvar_set ( global_id ( name ) , val ) ;
}
VALUE
rb_gvar_get ( ID id )
{
struct rb_global_entry * entry = rb_global_entry ( id ) ;
struct rb_global_variable * var = entry - > var ;
return ( * var - > getter ) ( entry - > id , var - > data ) ;
1998-01-16 15:13:05 +03:00
}
1999-11-17 10:30:37 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_gv_get ( const char * name )
1999-11-17 10:30:37 +03:00
{
2018-11-25 07:05:09 +03:00
ID id = find_global_id ( name ) ;
2020-07-20 07:31:41 +03:00
2018-11-25 07:05:09 +03:00
if ( ! id ) {
2024-01-19 10:03:38 +03:00
rb_warning ( " global variable '%s' not initialized " , name ) ;
2018-11-25 07:05:09 +03:00
return Qnil ;
}
2020-07-03 10:56:44 +03:00
return rb_gvar_get ( id ) ;
1999-11-17 10:30:37 +03:00
}
2023-03-07 08:34:31 +03:00
VALUE
2020-07-03 10:56:44 +03:00
rb_gvar_defined ( ID id )
1998-01-16 15:13:05 +03:00
{
2020-07-03 10:56:44 +03:00
struct rb_global_entry * entry = rb_global_entry ( id ) ;
2021-09-15 02:11:05 +03:00
return RBOOL ( entry - > var - > getter ! = rb_gvar_undef_getter ) ;
1998-01-16 15:13:05 +03:00
}
2018-09-11 12:48:58 +03:00
rb_gvar_getter_t *
2020-07-03 10:56:44 +03:00
rb_gvar_getter_function_of ( ID id )
2018-09-11 12:48:58 +03:00
{
2020-07-03 10:56:44 +03:00
const struct rb_global_entry * entry = rb_global_entry ( id ) ;
2018-09-11 12:48:58 +03:00
return entry - > var - > getter ;
}
rb_gvar_setter_t *
2020-07-03 10:56:44 +03:00
rb_gvar_setter_function_of ( ID id )
2018-09-11 12:48:58 +03:00
{
2020-07-03 10:56:44 +03:00
const struct rb_global_entry * entry = rb_global_entry ( id ) ;
2018-09-11 12:48:58 +03:00
return entry - > var - > setter ;
}
2015-11-03 00:26:26 +03:00
static enum rb_id_table_iterator_result
gvar_i ( ID key , VALUE val , void * a )
1998-01-16 15:19:22 +03:00
{
2012-03-31 02:40:54 +04:00
VALUE ary = ( VALUE ) a ;
2006-09-04 09:46:47 +04:00
rb_ary_push ( ary , ID2SYM ( key ) ) ;
2015-11-03 00:26:26 +03:00
return ID_TABLE_CONTINUE ;
1998-01-16 15:19:22 +03:00
}
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_f_global_variables ( void )
1998-01-16 15:19:22 +03:00
{
1999-01-20 07:59:39 +03:00
VALUE ary = rb_ary_new ( ) ;
2016-01-14 11:36:49 +03:00
VALUE sym , backref = rb_backref_get ( ) ;
1998-01-16 15:19:22 +03:00
2020-03-09 20:22:11 +03:00
if ( ! rb_ractor_main_p ( ) ) {
2020-12-21 12:06:28 +03:00
rb_raise ( rb_eRactorIsolationError , " can not access global variables from non-main Ractors " ) ;
2020-03-09 20:22:11 +03:00
}
rb_id_table_foreach ( rb_global_tbl , gvar_i , ( void * ) ary ) ;
2016-01-14 11:36:49 +03:00
if ( ! NIL_P ( backref ) ) {
2016-01-14 05:45:03 +03:00
char buf [ 2 ] ;
2016-01-14 11:36:49 +03:00
int i , nmatch = rb_match_count ( backref ) ;
2016-01-14 05:45:03 +03:00
buf [ 0 ] = ' $ ' ;
2016-01-14 11:36:49 +03:00
for ( i = 1 ; i < = nmatch ; + + i ) {
2023-06-27 16:39:15 +03:00
if ( ! RTEST ( rb_reg_nth_defined ( i , backref ) ) ) continue ;
2016-01-14 11:36:49 +03:00
if ( i < 10 ) {
/* probably reused, make static ID */
buf [ 1 ] = ( char ) ( i + ' 0 ' ) ;
sym = ID2SYM ( rb_intern2 ( buf , 2 ) ) ;
}
else {
/* dynamic symbol */
sym = rb_str_intern ( rb_sprintf ( " $%d " , i ) ) ;
}
rb_ary_push ( ary , sym ) ;
2016-01-14 05:45:03 +03:00
}
1998-01-16 15:19:22 +03:00
}
return ary ;
}
1998-01-16 15:13:05 +03:00
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_alias_variable ( ID name1 , ID name2 )
1998-01-16 15:13:05 +03:00
{
2015-11-30 02:53:43 +03:00
struct rb_global_entry * entry1 , * entry2 ;
2015-11-03 00:26:26 +03:00
VALUE data1 ;
2020-03-09 20:22:11 +03:00
struct rb_id_table * gtbl = rb_global_tbl ;
if ( ! rb_ractor_main_p ( ) ) {
2020-12-21 12:06:28 +03:00
rb_raise ( rb_eRactorIsolationError , " can not access global variables from non-main Ractors " ) ;
2020-03-09 20:22:11 +03:00
}
1998-01-16 15:13:05 +03:00
2001-10-22 20:20:14 +04:00
entry2 = rb_global_entry ( name2 ) ;
2020-07-03 10:56:44 +03:00
if ( ! rb_id_table_lookup ( gtbl , name1 , & data1 ) ) {
2015-11-30 02:53:43 +03:00
entry1 = ALLOC ( struct rb_global_entry ) ;
2001-10-22 20:20:14 +04:00
entry1 - > id = name1 ;
2020-07-03 10:56:44 +03:00
rb_id_table_insert ( gtbl , name1 , ( VALUE ) entry1 ) ;
2001-10-22 20:20:14 +04:00
}
2015-11-30 02:53:43 +03:00
else if ( ( entry1 = ( struct rb_global_entry * ) data1 ) - > var ! = entry2 - > var ) {
struct rb_global_variable * var = entry1 - > var ;
2001-10-22 20:20:14 +04:00
if ( var - > block_trace ) {
rb_raise ( rb_eRuntimeError , " can't alias in tracer " ) ;
}
var - > counter - - ;
if ( var - > counter = = 0 ) {
struct trace_var * trace = var - > trace ;
while ( trace ) {
struct trace_var * next = trace - > next ;
* array.c, bignum.c, cont.c, dir.c, dln.c, encoding.c, enumerator.c,
enumerator.c (enumerator_allocate), eval_jump.c, file.c, hash.c,
io.c, load.c, pack.c, proc.c, random.c, re.c, ruby.c, st.c,
string.c, thread.c, thread_pthread.c, time.c, util.c, variable.c,
vm.c, gc.c:
allocated memory objects by xmalloc (ruby_xmalloc) should be
freed by xfree (ruby_xfree).
* ext/curses/curses.c, ext/dbm/dbm.c, ext/digest/digest.c,
ext/gdbm/gdbm.c, ext/json/ext/parser/parser.c,
ext/json/ext/parser/unicode.c, ext/openssl/ossl_cipher.c,
ext/openssl/ossl_hmac.c, ext/openssl/ossl_pkey_ec.c,
ext/sdbm/init.c, ext/strscan/strscan.c, ext/zlib/zlib.c:
ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@17017 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2008-06-08 14:01:40 +04:00
xfree ( trace ) ;
2001-10-22 20:20:14 +04:00
trace = next ;
}
* array.c, bignum.c, cont.c, dir.c, dln.c, encoding.c, enumerator.c,
enumerator.c (enumerator_allocate), eval_jump.c, file.c, hash.c,
io.c, load.c, pack.c, proc.c, random.c, re.c, ruby.c, st.c,
string.c, thread.c, thread_pthread.c, time.c, util.c, variable.c,
vm.c, gc.c:
allocated memory objects by xmalloc (ruby_xmalloc) should be
freed by xfree (ruby_xfree).
* ext/curses/curses.c, ext/dbm/dbm.c, ext/digest/digest.c,
ext/gdbm/gdbm.c, ext/json/ext/parser/parser.c,
ext/json/ext/parser/unicode.c, ext/openssl/ossl_cipher.c,
ext/openssl/ossl_hmac.c, ext/openssl/ossl_pkey_ec.c,
ext/sdbm/init.c, ext/strscan/strscan.c, ext/zlib/zlib.c:
ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@17017 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2008-06-08 14:01:40 +04:00
xfree ( var ) ;
2001-10-22 20:20:14 +04:00
}
2001-10-17 09:28:02 +04:00
}
2001-10-22 20:20:14 +04:00
else {
return ;
2001-10-17 09:28:02 +04:00
}
2001-10-22 20:20:14 +04:00
entry2 - > var - > counter + + ;
entry1 - > var = entry2 - > var ;
1998-01-16 15:13:05 +03:00
}
2020-03-09 20:22:11 +03:00
static void
IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR ( ID id )
{
if ( UNLIKELY ( ! rb_ractor_main_p ( ) ) ) {
if ( rb_is_instance_id ( id ) ) { // check only normal ivars
2021-10-22 11:24:34 +03:00
rb_raise ( rb_eRactorIsolationError , " can not set instance variables of classes/modules by non-main Ractors " ) ;
2020-03-09 20:22:11 +03:00
}
}
}
# define CVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR() \
if ( UNLIKELY ( ! rb_ractor_main_p ( ) ) ) { \
2020-12-21 12:06:28 +03:00
rb_raise ( rb_eRactorIsolationError , " can not access class variables from non-main Ractors " ) ; \
2020-03-09 20:22:11 +03:00
}
static inline struct st_table *
generic_ivtbl ( VALUE obj , ID id , bool force_check_ractor )
{
2020-10-14 04:43:13 +03:00
ASSERT_vm_locking ( ) ;
2020-10-21 18:36:53 +03:00
if ( ( force_check_ractor | | LIKELY ( rb_is_instance_id ( id ) ) /* not internal ID */ ) & &
! RB_OBJ_FROZEN_RAW ( obj ) & &
2020-10-21 16:54:39 +03:00
UNLIKELY ( ! rb_ractor_main_p ( ) ) & &
UNLIKELY ( rb_ractor_shareable_p ( obj ) ) ) {
2020-12-11 10:37:20 +03:00
2020-12-21 12:06:28 +03:00
rb_raise ( rb_eRactorIsolationError , " can not access instance variables of shareable objects from non-main Ractors " ) ;
2020-03-09 20:22:11 +03:00
}
return generic_iv_tbl_ ;
}
static inline struct st_table *
generic_ivtbl_no_ractor_check ( VALUE obj )
{
return generic_ivtbl ( obj , 0 , false ) ;
}
2023-03-07 08:34:31 +03:00
int
2022-10-03 18:14:32 +03:00
rb_gen_ivtbl_get ( VALUE obj , ID id , struct gen_ivtbl * * ivtbl )
2015-05-30 02:42:49 +03:00
{
2022-11-01 00:05:37 +03:00
RUBY_ASSERT ( ! RB_TYPE_P ( obj , T_ICLASS ) ) ;
2015-05-30 02:42:49 +03:00
st_data_t data ;
2020-10-14 04:43:13 +03:00
int r = 0 ;
2015-05-30 02:42:49 +03:00
2020-10-14 04:43:13 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
if ( st_lookup ( generic_ivtbl ( obj , id , false ) , ( st_data_t ) obj , & data ) ) {
* ivtbl = ( struct gen_ivtbl * ) data ;
r = 1 ;
}
2015-05-30 02:42:49 +03:00
}
2020-10-14 04:43:13 +03:00
RB_VM_LOCK_LEAVE ( ) ;
return r ;
}
2023-03-07 08:34:31 +03:00
int
2020-10-14 04:43:13 +03:00
rb_ivar_generic_ivtbl_lookup ( VALUE obj , struct gen_ivtbl * * ivtbl )
{
2022-10-03 18:14:32 +03:00
return rb_gen_ivtbl_get ( obj , 0 , ivtbl ) ;
1999-01-20 07:59:39 +03:00
}
2015-05-30 02:42:49 +03:00
static size_t
gen_ivtbl_bytes ( size_t n )
2014-10-08 18:49:49 +04:00
{
2023-10-31 16:38:35 +03:00
return offsetof ( struct gen_ivtbl , as . shape . ivptr ) + n * sizeof ( VALUE ) ;
2015-05-30 02:42:49 +03:00
}
2014-10-08 18:49:49 +04:00
2015-12-08 18:07:41 +03:00
static struct gen_ivtbl *
2016-04-22 12:47:34 +03:00
gen_ivtbl_resize ( struct gen_ivtbl * old , uint32_t n )
2015-05-30 02:42:49 +03:00
{
2022-10-03 18:14:32 +03:00
RUBY_ASSERT ( n > 0 ) ;
2023-10-31 16:38:35 +03:00
uint32_t len = old ? old - > as . shape . numiv : 0 ;
2015-05-30 02:42:49 +03:00
struct gen_ivtbl * ivtbl = xrealloc ( old , gen_ivtbl_bytes ( n ) ) ;
2023-10-31 16:38:35 +03:00
ivtbl - > as . shape . numiv = n ;
2015-05-30 02:42:49 +03:00
for ( ; len < n ; len + + ) {
2023-10-31 16:38:35 +03:00
ivtbl - > as . shape . ivptr [ len ] = Qundef ;
2014-10-08 18:49:49 +04:00
}
2015-05-30 02:42:49 +03:00
return ivtbl ;
2014-10-08 18:49:49 +04:00
}
2015-05-30 02:42:49 +03:00
void
2023-11-24 18:25:06 +03:00
rb_mark_generic_ivar ( VALUE obj )
1999-08-13 09:45:20 +04:00
{
2023-10-31 16:38:35 +03:00
struct gen_ivtbl * ivtbl ;
2015-05-30 02:42:49 +03:00
2023-10-31 16:38:35 +03:00
if ( rb_gen_ivtbl_get ( obj , 0 , & ivtbl ) ) {
if ( rb_shape_obj_too_complex ( obj ) ) {
2023-11-24 18:25:06 +03:00
rb_mark_tbl_no_pin ( ivtbl - > as . complex . table ) ;
2023-10-20 02:01:35 +03:00
}
2023-10-31 16:38:35 +03:00
else {
for ( uint32_t i = 0 ; i < ivtbl - > as . shape . numiv ; i + + ) {
2023-11-24 18:25:06 +03:00
rb_gc_mark_movable ( ivtbl - > as . shape . ivptr [ i ] ) ;
}
}
}
}
void
rb_ref_update_generic_ivar ( VALUE obj )
{
struct gen_ivtbl * ivtbl ;
if ( rb_gen_ivtbl_get ( obj , 0 , & ivtbl ) ) {
if ( rb_shape_obj_too_complex ( obj ) ) {
rb_gc_ref_update_table_values_only ( ivtbl - > as . complex . table ) ;
}
else {
for ( uint32_t i = 0 ; i < ivtbl - > as . shape . numiv ; i + + ) {
ivtbl - > as . shape . ivptr [ i ] = rb_gc_location ( ivtbl - > as . shape . ivptr [ i ] ) ;
2023-10-31 16:38:35 +03:00
}
2023-10-20 02:01:35 +03:00
}
2015-05-30 02:42:49 +03:00
}
1999-08-13 09:45:20 +04:00
}
2019-04-20 04:19:47 +03:00
void
rb_mv_generic_ivar ( VALUE rsrc , VALUE dst )
{
st_data_t key = ( st_data_t ) rsrc ;
2021-06-23 04:50:09 +03:00
st_data_t ivtbl ;
2019-04-20 04:19:47 +03:00
2021-06-23 04:50:09 +03:00
if ( st_delete ( generic_ivtbl_no_ractor_check ( rsrc ) , & key , & ivtbl ) )
st_insert ( generic_ivtbl_no_ractor_check ( dst ) , ( st_data_t ) dst , ivtbl ) ;
2019-04-20 04:19:47 +03:00
}
1999-01-20 07:59:39 +03:00
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_free_generic_ivar ( VALUE obj )
1999-01-20 07:59:39 +03:00
{
2023-10-31 16:38:35 +03:00
st_data_t key = ( st_data_t ) obj , value ;
bool too_complex = rb_shape_obj_too_complex ( obj ) ;
1999-01-20 07:59:39 +03:00
2023-10-31 16:38:35 +03:00
if ( st_delete ( generic_ivtbl_no_ractor_check ( obj ) , & key , & value ) ) {
struct gen_ivtbl * ivtbl = ( struct gen_ivtbl * ) value ;
if ( UNLIKELY ( too_complex ) ) {
st_free_table ( ivtbl - > as . complex . table ) ;
}
xfree ( ivtbl ) ;
}
1999-01-20 07:59:39 +03:00
}
2024-02-23 00:02:10 +03:00
size_t
2009-06-22 18:25:04 +04:00
rb_generic_ivar_memsize ( VALUE obj )
2009-06-17 02:36:27 +04:00
{
2015-05-30 02:42:49 +03:00
struct gen_ivtbl * ivtbl ;
2023-10-31 16:38:35 +03:00
if ( rb_gen_ivtbl_get ( obj , 0 , & ivtbl ) ) {
if ( rb_shape_obj_too_complex ( obj ) ) {
return sizeof ( struct gen_ivtbl ) + st_memsize ( ivtbl - > as . complex . table ) ;
}
else {
return gen_ivtbl_bytes ( ivtbl - > as . shape . numiv ) ;
}
}
2009-06-17 02:36:27 +04:00
return 0 ;
}
2022-10-03 18:14:32 +03:00
# if !SHAPE_IN_BASIC_FLAGS
2023-03-07 08:34:31 +03:00
shape_id_t
2022-10-03 18:14:32 +03:00
rb_generic_shape_id ( VALUE obj )
{
struct gen_ivtbl * ivtbl = 0 ;
shape_id_t shape_id = 0 ;
RB_VM_LOCK_ENTER ( ) ;
{
st_table * global_iv_table = generic_ivtbl ( obj , 0 , false ) ;
if ( global_iv_table & & st_lookup ( global_iv_table , obj , ( st_data_t * ) & ivtbl ) ) {
shape_id = ivtbl - > shape_id ;
}
else if ( OBJ_FROZEN ( obj ) ) {
2022-11-08 23:35:31 +03:00
shape_id = SPECIAL_CONST_SHAPE_ID ;
2022-10-03 18:14:32 +03:00
}
}
RB_VM_LOCK_LEAVE ( ) ;
return shape_id ;
}
# endif
2015-05-30 02:42:49 +03:00
static size_t
2023-10-31 16:38:35 +03:00
gen_ivtbl_count ( VALUE obj , const struct gen_ivtbl * ivtbl )
1999-08-13 09:45:20 +04:00
{
2016-04-22 12:47:34 +03:00
uint32_t i ;
2015-05-30 02:42:49 +03:00
size_t n = 0 ;
1999-08-13 09:45:20 +04:00
2023-10-31 16:38:35 +03:00
if ( rb_shape_obj_too_complex ( obj ) ) {
n = st_table_size ( ivtbl - > as . complex . table ) ;
}
else {
for ( i = 0 ; i < ivtbl - > as . shape . numiv ; i + + ) {
if ( ! UNDEF_P ( ivtbl - > as . shape . ivptr [ i ] ) ) {
n + + ;
}
2002-09-03 09:20:14 +04:00
}
1999-08-13 09:45:20 +04:00
}
2015-05-30 02:42:49 +03:00
return n ;
1999-08-13 09:45:20 +04:00
}
2015-10-28 09:36:13 +03:00
VALUE
2013-05-02 12:30:56 +04:00
rb_ivar_lookup ( VALUE obj , ID id , VALUE undef )
1998-01-16 15:13:05 +03:00
{
variable.c: remove generic ivar support for special constants
Special constants are all frozen since [Feature #8923] and cannot
support ivars. Remove some unused code we had for supporting them.
* variable.c (special_generic_ivar): remove flag
(givar_i, rb_mark_generic_ivar_tbl): remove functions
(rb_free_generic_ivar, rb_ivar_lookup, rb_ivar_delete,
generic_ivar_set, rb_ivar_set, rb_ivar_defined,
rb_copy_generic_ivar, rb_ivar_foreach, rb_ivar_count,
rb_obj_remove_instance_variable):
adjust for lack of ivar support in special constants
* test/ruby/test_variable.rb: test ivars for special consts
* internal.h: remove rb_mark_generic_ivar_tbl decl
* gc.c (gc_mark_roots): remove rb_mark_generic_ivar_tbl call
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@50758 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-06-03 23:53:35 +03:00
if ( SPECIAL_CONST_P ( obj ) ) return undef ;
2022-10-03 18:14:32 +03:00
shape_id_t shape_id ;
VALUE * ivar_list ;
rb_shape_t * shape ;
# if SHAPE_IN_BASIC_FLAGS
shape_id = RBASIC_SHAPE_ID ( obj ) ;
# endif
2012-07-28 21:04:58 +04:00
switch ( BUILTIN_TYPE ( obj ) ) {
1998-01-16 15:13:05 +03:00
case T_CLASS :
case T_MODULE :
2020-10-16 09:20:40 +03:00
{
2023-10-19 21:00:54 +03:00
bool found = false ;
2022-11-01 00:05:37 +03:00
VALUE val ;
2021-06-23 04:50:09 +03:00
2022-11-01 00:05:37 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
# if !SHAPE_IN_BASIC_FLAGS
shape_id = RCLASS_SHAPE_ID ( obj ) ;
# endif
2023-10-19 21:00:54 +03:00
if ( rb_shape_obj_too_complex ( obj ) ) {
st_table * iv_table = RCLASS_IV_HASH ( obj ) ;
if ( rb_st_lookup ( iv_table , ( st_data_t ) id , ( st_data_t * ) & val ) ) {
found = true ;
}
else {
val = undef ;
}
2022-11-01 00:05:37 +03:00
}
else {
2023-10-19 21:00:54 +03:00
attr_index_t index = 0 ;
shape = rb_shape_get_shape_by_id ( shape_id ) ;
found = rb_shape_get_iv_index ( shape , id , & index ) ;
if ( found ) {
ivar_list = RCLASS_IVPTR ( obj ) ;
RUBY_ASSERT ( ivar_list ) ;
val = ivar_list [ index ] ;
}
else {
val = undef ;
}
2021-10-22 11:24:34 +03:00
}
2020-10-16 09:20:40 +03:00
}
2022-11-01 00:05:37 +03:00
RB_VM_LOCK_LEAVE ( ) ;
if ( found & &
rb_is_instance_id ( id ) & &
UNLIKELY ( ! rb_ractor_main_p ( ) ) & &
! rb_ractor_shareable_p ( val ) ) {
rb_raise ( rb_eRactorIsolationError ,
" can not get unshareable values from instance variables of classes/modules from non-main Ractors " ) ;
2020-10-16 09:20:40 +03:00
}
2022-11-01 00:05:37 +03:00
return val ;
2020-10-16 09:20:40 +03:00
}
2022-10-03 18:14:32 +03:00
case T_OBJECT :
{
# if !SHAPE_IN_BASIC_FLAGS
shape_id = ROBJECT_SHAPE_ID ( obj ) ;
# endif
2022-12-09 01:16:52 +03:00
if ( rb_shape_obj_too_complex ( obj ) ) {
2023-03-17 21:29:04 +03:00
st_table * iv_table = ROBJECT_IV_HASH ( obj ) ;
2022-12-09 01:16:52 +03:00
VALUE val ;
2023-03-17 21:29:04 +03:00
if ( rb_st_lookup ( iv_table , ( st_data_t ) id , ( st_data_t * ) & val ) ) {
2022-12-09 01:16:52 +03:00
return val ;
}
else {
return undef ;
}
}
RUBY_ASSERT ( ! rb_shape_obj_too_complex ( obj ) ) ;
2022-10-03 18:14:32 +03:00
ivar_list = ROBJECT_IVPTR ( obj ) ;
break ;
}
1998-01-16 15:13:05 +03:00
default :
2022-10-03 18:14:32 +03:00
if ( FL_TEST_RAW ( obj , FL_EXIVAR ) ) {
struct gen_ivtbl * ivtbl ;
rb_gen_ivtbl_get ( obj , id , & ivtbl ) ;
2023-10-20 02:01:35 +03:00
if ( rb_shape_obj_too_complex ( obj ) ) {
VALUE val ;
2023-10-31 16:38:35 +03:00
if ( rb_st_lookup ( ivtbl - > as . complex . table , ( st_data_t ) id , ( st_data_t * ) & val ) ) {
2023-10-20 02:01:35 +03:00
return val ;
}
else {
return undef ;
}
}
2022-10-03 18:14:32 +03:00
# if !SHAPE_IN_BASIC_FLAGS
shape_id = ivtbl - > shape_id ;
# endif
2023-10-31 16:38:35 +03:00
ivar_list = ivtbl - > as . shape . ivptr ;
2022-11-10 03:11:20 +03:00
}
else {
2022-10-03 18:14:32 +03:00
return undef ;
}
1998-01-16 15:13:05 +03:00
break ;
}
2022-10-03 18:14:32 +03:00
attr_index_t index = 0 ;
shape = rb_shape_get_shape_by_id ( shape_id ) ;
if ( rb_shape_get_iv_index ( shape , id , & index ) ) {
return ivar_list [ index ] ;
}
2013-05-02 12:30:56 +04:00
return undef ;
1998-01-16 15:13:05 +03:00
}
2003-02-18 17:30:17 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_ivar_get ( VALUE obj , ID id )
2003-02-18 17:30:17 +03:00
{
2020-12-10 07:53:02 +03:00
VALUE iv = rb_ivar_lookup ( obj , id , Qnil ) ;
2017-03-15 11:25:58 +03:00
RB_DEBUG_COUNTER_INC ( ivar_get_base ) ;
2013-05-02 12:30:56 +04:00
return iv ;
2003-02-18 17:30:17 +03:00
}
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_attr_get ( VALUE obj , ID id )
2003-02-18 17:30:17 +03:00
{
2013-05-02 12:30:56 +04:00
return rb_ivar_lookup ( obj , id , Qnil ) ;
2003-02-18 17:30:17 +03:00
}
2015-05-30 03:20:15 +03:00
static VALUE
rb_ivar_delete ( VALUE obj , ID id , VALUE undef )
{
2022-10-01 02:01:50 +03:00
rb_check_frozen ( obj ) ;
2022-11-01 00:05:37 +03:00
VALUE val = undef ;
2023-11-01 20:13:51 +03:00
rb_shape_t * shape = rb_shape_get_shape ( obj ) ;
2022-10-03 18:14:32 +03:00
2023-11-01 20:13:51 +03:00
if ( BUILTIN_TYPE ( obj ) = = T_CLASS | | BUILTIN_TYPE ( obj ) = = T_MODULE ) {
2020-03-09 20:22:11 +03:00
IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR ( id ) ;
2023-11-01 20:13:51 +03:00
}
2022-11-01 00:05:37 +03:00
2023-11-01 20:13:51 +03:00
if ( ! rb_shape_transition_shape_remove_ivar ( obj , id , shape , & val ) ) {
if ( ! rb_shape_obj_too_complex ( obj ) ) {
2023-11-16 18:28:12 +03:00
rb_evict_ivars_to_hash ( obj ) ;
2020-10-16 09:20:40 +03:00
}
2022-11-01 00:05:37 +03:00
2023-11-01 20:13:51 +03:00
st_table * table = NULL ;
switch ( BUILTIN_TYPE ( obj ) ) {
case T_CLASS :
case T_MODULE :
table = RCLASS_IV_HASH ( obj ) ;
break ;
2022-10-03 18:14:32 +03:00
2023-11-01 20:13:51 +03:00
case T_OBJECT :
table = ROBJECT_IV_HASH ( obj ) ;
break ;
default : {
struct gen_ivtbl * ivtbl ;
if ( rb_gen_ivtbl_get ( obj , 0 , & ivtbl ) ) {
table = ivtbl - > as . complex . table ;
}
break ;
}
}
if ( table ) {
if ( ! st_delete ( table , ( st_data_t * ) & id , ( st_data_t * ) & val ) ) {
val = undef ;
}
}
2015-05-30 03:20:15 +03:00
}
2022-10-03 18:14:32 +03:00
2022-11-01 00:05:37 +03:00
return val ;
2015-05-30 03:20:15 +03:00
}
VALUE
rb_attr_delete ( VALUE obj , ID id )
{
return rb_ivar_delete ( obj , id , Qnil ) ;
}
2023-10-20 02:01:35 +03:00
void
2023-11-16 19:50:21 +03:00
rb_obj_convert_to_too_complex ( VALUE obj , st_table * table )
2023-10-20 02:01:35 +03:00
{
RUBY_ASSERT ( ! rb_shape_obj_too_complex ( obj ) ) ;
2023-11-16 19:50:21 +03:00
VALUE * old_ivptr = NULL ;
2023-10-20 02:01:35 +03:00
switch ( BUILTIN_TYPE ( obj ) ) {
2023-11-20 21:43:31 +03:00
case T_OBJECT :
if ( ! ( RBASIC ( obj ) - > flags & ROBJECT_EMBED ) ) {
old_ivptr = ROBJECT_IVPTR ( obj ) ;
}
rb_shape_set_shape_id ( obj , OBJ_TOO_COMPLEX_SHAPE_ID ) ;
ROBJECT_SET_IV_HASH ( obj , table ) ;
break ;
case T_CLASS :
case T_MODULE :
old_ivptr = RCLASS_IVPTR ( obj ) ;
rb_shape_set_shape_id ( obj , OBJ_TOO_COMPLEX_SHAPE_ID ) ;
RCLASS_SET_IV_HASH ( obj , table ) ;
break ;
default :
RB_VM_LOCK_ENTER ( ) ;
{
struct st_table * gen_ivs = generic_ivtbl_no_ractor_check ( obj ) ;
2023-11-22 18:04:57 +03:00
struct gen_ivtbl * old_ivtbl = NULL ;
st_lookup ( gen_ivs , ( st_data_t ) obj , ( st_data_t * ) & old_ivtbl ) ;
if ( old_ivtbl ) {
/* We need to modify old_ivtbl to have the too complex shape
* and hold the table because the xmalloc could trigger a GC
2024-03-05 12:50:47 +03:00
* compaction . We want the table to be updated rather than
2023-11-22 18:04:57 +03:00
* the original ivptr . */
# if SHAPE_IN_BASIC_FLAGS
rb_shape_set_shape_id ( obj , OBJ_TOO_COMPLEX_SHAPE_ID ) ;
# else
old_ivtbl - > shape_id = OBJ_TOO_COMPLEX_SHAPE_ID ;
# endif
old_ivtbl - > as . complex . table = table ;
old_ivptr = ( VALUE * ) old_ivtbl ;
}
2023-11-20 21:43:31 +03:00
struct gen_ivtbl * ivtbl = xmalloc ( sizeof ( struct gen_ivtbl ) ) ;
ivtbl - > as . complex . table = table ;
2023-11-20 22:55:50 +03:00
st_insert ( gen_ivs , ( st_data_t ) obj , ( st_data_t ) ivtbl ) ;
2023-10-31 16:38:35 +03:00
# if SHAPE_IN_BASIC_FLAGS
2023-11-20 21:43:31 +03:00
rb_shape_set_shape_id ( obj , OBJ_TOO_COMPLEX_SHAPE_ID ) ;
2023-10-31 16:38:35 +03:00
# else
2023-11-20 21:43:31 +03:00
ivtbl - > shape_id = OBJ_TOO_COMPLEX_SHAPE_ID ;
2023-10-31 16:38:35 +03:00
# endif
2023-11-20 21:43:31 +03:00
}
RB_VM_LOCK_LEAVE ( ) ;
2023-11-16 19:50:21 +03:00
}
2023-10-20 02:01:35 +03:00
2024-01-17 23:55:08 +03:00
xfree ( old_ivptr ) ;
2023-11-16 19:50:21 +03:00
}
void
rb_evict_ivars_to_hash ( VALUE obj )
{
RUBY_ASSERT ( ! rb_shape_obj_too_complex ( obj ) ) ;
st_table * table = st_init_numtable_with_size ( rb_ivar_count ( obj ) ) ;
// Evacuate all previous values from shape into id_table
rb_obj_copy_ivs_to_hash_table ( obj , table ) ;
rb_obj_convert_to_too_complex ( obj , table ) ;
2023-10-31 16:38:35 +03:00
RUBY_ASSERT ( rb_shape_obj_too_complex ( obj ) ) ;
2023-10-20 02:01:35 +03:00
}
2023-11-06 16:28:20 +03:00
struct general_ivar_set_result {
attr_index_t index ;
bool existing ;
} ;
static struct general_ivar_set_result
general_ivar_set ( VALUE obj , ID id , VALUE val , void * data ,
VALUE * ( * shape_ivptr_func ) ( VALUE , void * ) ,
void ( * shape_resize_ivptr_func ) ( VALUE , attr_index_t , attr_index_t , void * ) ,
void ( * set_shape_func ) ( VALUE , rb_shape_t * , void * ) ,
void ( * transition_too_complex_func ) ( VALUE , void * ) ,
st_table * ( * too_complex_table_func ) ( VALUE , void * ) )
{
struct general_ivar_set_result result = {
. index = 0 ,
. existing = true
} ;
2023-11-07 20:09:55 +03:00
rb_shape_t * current_shape = rb_shape_get_shape ( obj ) ;
2023-11-06 16:28:20 +03:00
2023-11-07 20:09:55 +03:00
if ( UNLIKELY ( current_shape - > type = = SHAPE_OBJ_TOO_COMPLEX ) ) {
2023-11-06 16:28:20 +03:00
goto too_complex ;
}
attr_index_t index ;
2023-11-07 20:09:55 +03:00
if ( ! rb_shape_get_iv_index ( current_shape , id , & index ) ) {
2023-11-06 16:28:20 +03:00
result . existing = false ;
2023-11-07 20:09:55 +03:00
index = current_shape - > next_iv_index ;
2023-11-06 16:28:20 +03:00
if ( index > = MAX_IVARS ) {
rb_raise ( rb_eArgError , " too many instance variables " ) ;
}
2023-11-07 20:09:55 +03:00
rb_shape_t * next_shape = rb_shape_get_next ( current_shape , obj , id ) ;
if ( UNLIKELY ( next_shape - > type = = SHAPE_OBJ_TOO_COMPLEX ) ) {
2023-11-06 16:28:20 +03:00
transition_too_complex_func ( obj , data ) ;
goto too_complex ;
}
2023-11-07 20:09:55 +03:00
else if ( UNLIKELY ( next_shape - > capacity ! = current_shape - > capacity ) ) {
RUBY_ASSERT ( next_shape - > capacity > current_shape - > capacity ) ;
shape_resize_ivptr_func ( obj , current_shape - > capacity , next_shape - > capacity , data ) ;
}
2023-11-06 16:28:20 +03:00
2023-11-07 20:09:55 +03:00
RUBY_ASSERT ( next_shape - > type = = SHAPE_IVAR ) ;
RUBY_ASSERT ( index = = ( next_shape - > next_iv_index - 1 ) ) ;
set_shape_func ( obj , next_shape , data ) ;
2023-11-06 16:28:20 +03:00
}
VALUE * table = shape_ivptr_func ( obj , data ) ;
RB_OBJ_WRITE ( obj , & table [ index ] , val ) ;
result . index = index ;
return result ;
too_complex :
{
RUBY_ASSERT ( rb_shape_obj_too_complex ( obj ) ) ;
st_table * table = too_complex_table_func ( obj , data ) ;
result . existing = st_insert ( table , ( st_data_t ) id , ( st_data_t ) val ) ;
result . index = 0 ;
RB_OBJ_WRITTEN ( obj , Qundef , val ) ;
}
return result ;
}
2023-11-03 15:53:50 +03:00
struct gen_ivar_lookup_ensure_size {
2023-11-06 16:34:38 +03:00
VALUE obj ;
ID id ;
2023-11-03 15:53:50 +03:00
struct gen_ivtbl * ivtbl ;
rb_shape_t * shape ;
bool resize ;
} ;
static int
generic_ivar_lookup_ensure_size ( st_data_t * k , st_data_t * v , st_data_t u , int existing )
2015-05-30 02:22:40 +03:00
{
2023-11-03 15:53:50 +03:00
ASSERT_vm_locking ( ) ;
2022-11-17 17:47:18 +03:00
2023-11-03 15:53:50 +03:00
struct gen_ivar_lookup_ensure_size * ivar_lookup = ( struct gen_ivar_lookup_ensure_size * ) u ;
struct gen_ivtbl * ivtbl = existing ? ( struct gen_ivtbl * ) * v : NULL ;
2023-10-30 19:50:56 +03:00
2023-11-03 15:53:50 +03:00
if ( ! existing | | ivar_lookup - > resize ) {
if ( existing ) {
RUBY_ASSERT ( ivar_lookup - > shape - > type = = SHAPE_IVAR ) ;
2023-11-11 00:17:39 +03:00
RUBY_ASSERT ( rb_shape_get_shape_by_id ( ivar_lookup - > shape - > parent_id ) - > capacity < ivar_lookup - > shape - > capacity ) ;
2023-11-03 15:53:50 +03:00
}
else {
FL_SET_RAW ( ( VALUE ) * k , FL_EXIVAR ) ;
}
ivtbl = gen_ivtbl_resize ( ivtbl , ivar_lookup - > shape - > capacity ) ;
* v = ( st_data_t ) ivtbl ;
}
RUBY_ASSERT ( FL_TEST ( ( VALUE ) * k , FL_EXIVAR ) ) ;
ivar_lookup - > ivtbl = ivtbl ;
2023-11-06 16:34:38 +03:00
if ( ivar_lookup - > shape ) {
# if SHAPE_IN_BASIC_FLAGS
rb_shape_set_shape ( ivar_lookup - > obj , ivar_lookup - > shape ) ;
# else
ivtbl - > shape_id = rb_shape_id ( ivar_lookup - > shape ) ;
# endif
}
2023-11-03 15:53:50 +03:00
return ST_CONTINUE ;
}
2023-11-06 16:34:38 +03:00
static VALUE *
generic_ivar_set_shape_ivptr ( VALUE obj , void * data )
2023-11-03 15:53:50 +03:00
{
2023-11-06 16:34:38 +03:00
RUBY_ASSERT ( ! rb_shape_obj_too_complex ( obj ) ) ;
struct gen_ivar_lookup_ensure_size * ivar_lookup = data ;
RB_VM_LOCK_ENTER ( ) ;
{
st_update ( generic_ivtbl ( obj , ivar_lookup - > id , false ) , ( st_data_t ) obj , generic_ivar_lookup_ensure_size , ( st_data_t ) ivar_lookup ) ;
2023-10-30 16:55:13 +03:00
}
2023-11-06 16:34:38 +03:00
RB_VM_LOCK_LEAVE ( ) ;
2023-10-30 16:55:13 +03:00
2023-11-06 16:34:38 +03:00
FL_SET_RAW ( obj , FL_EXIVAR ) ;
2023-11-03 15:53:50 +03:00
2023-11-06 16:34:38 +03:00
return ivar_lookup - > ivtbl - > as . shape . ivptr ;
}
2023-11-03 15:53:50 +03:00
2023-11-06 16:34:38 +03:00
static void
generic_ivar_set_shape_resize_ivptr ( VALUE obj , attr_index_t _old_capa , attr_index_t new_capa , void * data )
{
struct gen_ivar_lookup_ensure_size * ivar_lookup = data ;
2023-11-03 15:53:50 +03:00
2023-11-06 16:34:38 +03:00
ivar_lookup - > resize = true ;
}
2023-11-03 15:53:50 +03:00
2023-11-06 16:34:38 +03:00
static void
generic_ivar_set_set_shape ( VALUE obj , rb_shape_t * shape , void * data )
{
struct gen_ivar_lookup_ensure_size * ivar_lookup = data ;
2023-11-03 15:53:50 +03:00
2023-11-06 16:34:38 +03:00
ivar_lookup - > shape = shape ;
}
2023-11-03 15:53:50 +03:00
2023-11-06 16:34:38 +03:00
static void
generic_ivar_set_transition_too_complex ( VALUE obj , void * _data )
{
2023-11-16 18:28:12 +03:00
rb_evict_ivars_to_hash ( obj ) ;
2023-11-06 16:34:38 +03:00
FL_SET_RAW ( obj , FL_EXIVAR ) ;
}
2023-11-03 15:53:50 +03:00
2023-11-06 16:34:38 +03:00
static st_table *
generic_ivar_set_too_complex_table ( VALUE obj , void * data )
{
struct gen_ivar_lookup_ensure_size * ivar_lookup = data ;
struct gen_ivtbl * ivtbl ;
if ( ! rb_gen_ivtbl_get ( obj , 0 , & ivtbl ) ) {
ivtbl = xmalloc ( sizeof ( struct gen_ivtbl ) ) ;
# if !SHAPE_IN_BASIC_FLAGS
ivtbl - > shape_id = SHAPE_OBJ_TOO_COMPLEX ;
# endif
ivtbl - > as . complex . table = st_init_numtable_with_size ( 1 ) ;
RB_VM_LOCK_ENTER ( ) ;
{
st_insert ( generic_ivtbl ( obj , ivar_lookup - > id , false ) , ( st_data_t ) obj , ( st_data_t ) ivtbl ) ;
2023-10-20 02:01:35 +03:00
}
2023-11-06 16:34:38 +03:00
RB_VM_LOCK_LEAVE ( ) ;
2023-11-03 15:53:50 +03:00
2023-11-06 16:34:38 +03:00
FL_SET_RAW ( obj , FL_EXIVAR ) ;
2022-11-17 17:47:18 +03:00
}
2023-11-06 16:34:38 +03:00
RUBY_ASSERT ( rb_shape_obj_too_complex ( obj ) ) ;
2020-10-14 04:43:13 +03:00
2023-11-06 16:34:38 +03:00
return ivtbl - > as . complex . table ;
}
2015-05-30 02:42:49 +03:00
2023-11-06 16:34:38 +03:00
static void
generic_ivar_set ( VALUE obj , ID id , VALUE val )
{
struct gen_ivar_lookup_ensure_size ivar_lookup = {
. obj = obj ,
. id = id ,
. resize = false ,
. shape = NULL ,
} ;
2022-11-17 17:47:18 +03:00
2023-11-06 16:34:38 +03:00
general_ivar_set ( obj , id , val , & ivar_lookup ,
generic_ivar_set_shape_ivptr ,
generic_ivar_set_shape_resize_ivptr ,
generic_ivar_set_set_shape ,
generic_ivar_set_transition_too_complex ,
generic_ivar_set_too_complex_table ) ;
2015-05-30 02:22:40 +03:00
}
2022-10-03 18:14:32 +03:00
void
2022-11-08 23:35:31 +03:00
rb_ensure_iv_list_size ( VALUE obj , uint32_t current_capacity , uint32_t new_capacity )
2020-11-06 21:11:20 +03:00
{
2022-12-09 01:16:52 +03:00
RUBY_ASSERT ( ! rb_shape_obj_too_complex ( obj ) ) ;
2020-11-06 21:11:20 +03:00
if ( RBASIC ( obj ) - > flags & ROBJECT_EMBED ) {
2023-08-21 16:13:36 +03:00
VALUE * ptr = ROBJECT_IVPTR ( obj ) ;
VALUE * newptr = ALLOC_N ( VALUE , new_capacity ) ;
2022-11-08 23:35:31 +03:00
MEMCPY ( newptr , ptr , VALUE , current_capacity ) ;
2022-07-11 17:09:39 +03:00
RB_FL_UNSET_RAW ( obj , ROBJECT_EMBED ) ;
2020-11-06 21:11:20 +03:00
ROBJECT ( obj ) - > as . heap . ivptr = newptr ;
2021-06-16 16:07:05 +03:00
}
else {
2023-08-21 16:13:36 +03:00
REALLOC_N ( ROBJECT ( obj ) - > as . heap . ivptr , VALUE , new_capacity ) ;
2020-11-06 21:11:20 +03:00
}
2022-09-27 02:09:50 +03:00
}
2023-12-04 17:58:41 +03:00
static int
2023-11-16 19:50:21 +03:00
rb_obj_copy_ivs_to_hash_table_i ( ID key , VALUE val , st_data_t arg )
2022-12-09 01:16:52 +03:00
{
2023-11-22 18:04:57 +03:00
RUBY_ASSERT ( ! st_lookup ( ( st_table * ) arg , ( st_data_t ) key , NULL ) ) ;
st_add_direct ( ( st_table * ) arg , ( st_data_t ) key , ( st_data_t ) val ) ;
2022-12-09 01:16:52 +03:00
return ST_CONTINUE ;
}
2023-11-16 19:50:21 +03:00
void
rb_obj_copy_ivs_to_hash_table ( VALUE obj , st_table * table )
{
rb_ivar_foreach ( obj , rb_obj_copy_ivs_to_hash_table_i , ( st_data_t ) table ) ;
}
2023-11-06 16:29:41 +03:00
static VALUE *
obj_ivar_set_shape_ivptr ( VALUE obj , void * _data )
2021-10-19 00:01:40 +03:00
{
2023-11-06 16:29:41 +03:00
RUBY_ASSERT ( ! rb_shape_obj_too_complex ( obj ) ) ;
2022-10-03 18:14:32 +03:00
2023-11-06 16:29:41 +03:00
return ROBJECT_IVPTR ( obj ) ;
}
2022-12-09 01:16:52 +03:00
2023-11-06 16:29:41 +03:00
static void
obj_ivar_set_shape_resize_ivptr ( VALUE obj , attr_index_t old_capa , attr_index_t new_capa , void * _data )
{
rb_ensure_iv_list_size ( obj , old_capa , new_capa ) ;
}
2022-11-08 23:35:31 +03:00
2023-11-06 16:29:41 +03:00
static void
obj_ivar_set_set_shape ( VALUE obj , rb_shape_t * shape , void * _data )
{
rb_shape_set_shape ( obj , shape ) ;
}
2022-11-15 18:52:39 +03:00
2023-11-06 16:29:41 +03:00
static void
obj_ivar_set_transition_too_complex ( VALUE obj , void * _data )
{
2023-11-16 18:28:12 +03:00
rb_evict_ivars_to_hash ( obj ) ;
2023-11-06 16:29:41 +03:00
}
2022-11-15 18:52:39 +03:00
2023-11-06 16:29:41 +03:00
static st_table *
obj_ivar_set_too_complex_table ( VALUE obj , void * _data )
{
RUBY_ASSERT ( rb_shape_obj_too_complex ( obj ) ) ;
2021-10-19 00:01:40 +03:00
2023-11-06 16:29:41 +03:00
return ROBJECT_IV_HASH ( obj ) ;
}
2022-10-15 19:37:44 +03:00
2023-11-06 16:29:41 +03:00
attr_index_t
rb_obj_ivar_set ( VALUE obj , ID id , VALUE val )
{
return general_ivar_set ( obj , id , val , NULL ,
obj_ivar_set_shape_ivptr ,
obj_ivar_set_shape_resize_ivptr ,
obj_ivar_set_set_shape ,
obj_ivar_set_transition_too_complex ,
obj_ivar_set_too_complex_table ) . index ;
2022-10-03 18:14:32 +03:00
}
2022-09-23 20:54:42 +03:00
2022-10-03 18:14:32 +03:00
/* Set the instance variable +val+ on object +obj+ at ivar name +id+.
* This function only works with T_OBJECT objects , so make sure
* + obj + is of type T_OBJECT before using this function .
*/
VALUE
rb_vm_set_ivar_id ( VALUE obj , ID id , VALUE val )
{
2024-05-30 15:55:32 +03:00
rb_check_frozen ( obj ) ;
2022-11-15 18:52:39 +03:00
rb_obj_ivar_set ( obj , id , val ) ;
2022-10-03 18:14:32 +03:00
return val ;
}
bool
rb_shape_set_shape_id ( VALUE obj , shape_id_t shape_id )
{
if ( rb_shape_get_shape_id ( obj ) = = shape_id ) {
return false ;
2022-09-23 20:54:42 +03:00
}
2022-10-03 18:14:32 +03:00
# if SHAPE_IN_BASIC_FLAGS
RBASIC_SET_SHAPE_ID ( obj , shape_id ) ;
# else
switch ( BUILTIN_TYPE ( obj ) ) {
case T_OBJECT :
2022-10-24 11:42:29 +03:00
ROBJECT_SET_SHAPE_ID ( obj , shape_id ) ;
break ;
2022-10-03 18:14:32 +03:00
case T_CLASS :
case T_MODULE :
2023-04-14 23:27:37 +03:00
RCLASS_SET_SHAPE_ID ( obj , shape_id ) ;
2022-10-24 11:42:29 +03:00
break ;
2022-10-03 18:14:32 +03:00
default :
2022-11-08 23:35:31 +03:00
if ( shape_id ! = SPECIAL_CONST_SHAPE_ID ) {
2022-10-24 11:42:29 +03:00
struct gen_ivtbl * ivtbl = 0 ;
RB_VM_LOCK_ENTER ( ) ;
{
st_table * global_iv_table = generic_ivtbl ( obj , 0 , false ) ;
if ( st_lookup ( global_iv_table , obj , ( st_data_t * ) & ivtbl ) ) {
ivtbl - > shape_id = shape_id ;
}
else {
rb_bug ( " Expected shape_id entry in global iv table " ) ;
}
}
RB_VM_LOCK_LEAVE ( ) ;
}
2022-10-03 18:14:32 +03:00
}
# endif
return true ;
}
/**
* Prevents further modifications to the given object . : : rb_eFrozenError shall
* be raised if modification is attempted .
*
* @ param [ out ] x Object in question .
*/
void rb_obj_freeze_inline ( VALUE x )
{
if ( RB_FL_ABLE ( x ) ) {
2024-04-16 15:23:45 +03:00
RB_FL_SET_RAW ( x , RUBY_FL_FREEZE ) ;
if ( TYPE ( x ) = = T_STRING ) {
RB_FL_UNSET_RAW ( x , FL_USER3 ) ; // STR_CHILLED
}
2022-10-03 18:14:32 +03:00
2023-10-20 02:01:35 +03:00
rb_shape_t * next_shape = rb_shape_transition_shape_frozen ( x ) ;
// If we're transitioning from "not complex" to "too complex"
// then evict ivars. This can happen if we run out of shapes
if ( ! rb_shape_obj_too_complex ( x ) & & next_shape - > type = = SHAPE_OBJ_TOO_COMPLEX ) {
2023-11-16 18:28:12 +03:00
rb_evict_ivars_to_hash ( x ) ;
2023-10-20 02:01:35 +03:00
}
rb_shape_set_shape ( x , next_shape ) ;
2022-10-03 18:14:32 +03:00
2024-03-06 19:04:22 +03:00
if ( RBASIC_CLASS ( x ) ) {
2022-10-03 18:14:32 +03:00
rb_freeze_singleton_class ( x ) ;
}
}
2018-10-31 01:01:17 +03:00
}
2019-01-16 13:48:30 +03:00
static void
ivar_set ( VALUE obj , ID id , VALUE val )
2018-10-31 01:01:17 +03:00
{
2017-03-15 11:25:58 +03:00
RB_DEBUG_COUNTER_INC ( ivar_set_base ) ;
2017-02-21 11:18:15 +03:00
2012-07-28 21:04:58 +04:00
switch ( BUILTIN_TYPE ( obj ) ) {
1998-01-16 15:13:05 +03:00
case T_OBJECT :
2022-10-03 18:14:32 +03:00
{
2023-11-28 16:12:35 +03:00
rb_obj_ivar_set ( obj , id , val ) ;
break ;
2022-10-03 18:14:32 +03:00
}
1998-01-16 15:13:05 +03:00
case T_CLASS :
case T_MODULE :
2020-03-09 20:22:11 +03:00
IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR ( id ) ;
2018-10-31 01:01:17 +03:00
rb_class_ivar_set ( obj , id , val ) ;
2022-11-01 00:05:37 +03:00
2007-09-29 05:21:15 +04:00
break ;
1998-01-16 15:13:05 +03:00
default :
2018-10-31 01:01:17 +03:00
generic_ivar_set ( obj , id , val ) ;
break ;
1998-01-16 15:13:05 +03:00
}
2019-01-16 13:48:30 +03:00
}
VALUE
rb_ivar_set ( VALUE obj , ID id , VALUE val )
{
rb_check_frozen ( obj ) ;
ivar_set ( obj , id , val ) ;
1998-01-16 15:13:05 +03:00
return val ;
}
2019-01-16 13:48:30 +03:00
void
rb_ivar_set_internal ( VALUE obj , ID id , VALUE val )
{
// should be internal instance variable name (no @ prefix)
VM_ASSERT ( ! rb_is_instance_id ( id ) ) ;
ivar_set ( obj , id , val ) ;
}
1998-01-16 15:13:05 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_ivar_defined ( VALUE obj , ID id )
1998-01-16 15:13:05 +03:00
{
2022-10-03 18:14:32 +03:00
attr_index_t index ;
variable.c: remove generic ivar support for special constants
Special constants are all frozen since [Feature #8923] and cannot
support ivars. Remove some unused code we had for supporting them.
* variable.c (special_generic_ivar): remove flag
(givar_i, rb_mark_generic_ivar_tbl): remove functions
(rb_free_generic_ivar, rb_ivar_lookup, rb_ivar_delete,
generic_ivar_set, rb_ivar_set, rb_ivar_defined,
rb_copy_generic_ivar, rb_ivar_foreach, rb_ivar_count,
rb_obj_remove_instance_variable):
adjust for lack of ivar support in special constants
* test/ruby/test_variable.rb: test ivars for special consts
* internal.h: remove rb_mark_generic_ivar_tbl decl
* gc.c (gc_mark_roots): remove rb_mark_generic_ivar_tbl call
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@50758 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-06-03 23:53:35 +03:00
if ( SPECIAL_CONST_P ( obj ) ) return Qfalse ;
2022-12-09 01:16:52 +03:00
if ( rb_shape_obj_too_complex ( obj ) ) {
VALUE idx ;
2023-11-03 12:42:20 +03:00
st_table * table = NULL ;
switch ( BUILTIN_TYPE ( obj ) ) {
case T_CLASS :
case T_MODULE :
table = ( st_table * ) RCLASS_IVPTR ( obj ) ;
break ;
case T_OBJECT :
table = ROBJECT_IV_HASH ( obj ) ;
break ;
default : {
struct gen_ivtbl * ivtbl ;
if ( rb_gen_ivtbl_get ( obj , 0 , & ivtbl ) ) {
table = ivtbl - > as . complex . table ;
}
break ;
}
}
if ( ! table | | ! rb_st_lookup ( table , id , & idx ) ) {
2022-12-09 01:16:52 +03:00
return Qfalse ;
}
return Qtrue ;
}
else {
return RBOOL ( rb_shape_get_iv_index ( rb_shape_get_shape ( obj ) , id , & index ) ) ;
}
1998-01-16 15:13:05 +03:00
}
2019-08-27 06:08:32 +03:00
typedef int rb_ivar_foreach_callback_func ( ID key , VALUE val , st_data_t arg ) ;
2020-10-16 09:20:40 +03:00
st_data_t rb_st_nth_key ( st_table * tab , st_index_t index ) ;
2019-08-27 06:08:32 +03:00
Always lookup IV buffers when iterating
Always look up instance variable buffers when iterating. It is possible
for the instance variable buffer to change out from under the object
during iteration, so we cannot cache the buffer on the stack.
In the case of Bug #19095, the transient heap moved the buffer during
iteration:
```
Watchpoint 1 hit:
old value: 0x0000000107c00df8
new value: 0x00000001032743c0
Process 31720 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
1358 }
1359 MEMCPY(new_ptr, old_ptr, VALUE, len);
1360 ROBJECT(obj)->as.heap.ivptr = new_ptr;
-> 1361 }
1362 }
1363 #endif
1364
miniruby`rb_obj_transient_heap_evacuate:
-> 0x1006e5178 <+328>: b 0x1006e517c ; <+332> at variable.c:1362:1
0x1006e517c <+332>: ldp x29, x30, [sp, #0x50]
0x1006e5180 <+336>: add sp, sp, #0x60
0x1006e5184 <+340>: ret
Target 0: (miniruby) stopped.
(lldb) bt
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
* frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
frame #1: 0x00000001006cb150 miniruby`transient_heap_block_evacuate(theap=0x0000000100b196c0, block=0x0000000107c00000) at transient_heap.c:734:17
frame #2: 0x00000001006c854c miniruby`transient_heap_evacuate(dmy=0x0000000000000000) at transient_heap.c:808:17
frame #3: 0x00000001007fe6c0 miniruby`rb_postponed_job_flush(vm=0x0000000104402900) at vm_trace.c:1773:21
frame #4: 0x0000000100637a84 miniruby`rb_threadptr_execute_interrupts(th=0x0000000103803bc0, blocking_timing=0) at thread.c:2316:13
frame #5: 0x000000010078b730 miniruby`rb_vm_check_ints(ec=0x00000001048038d0) at vm_core.h:2025:9
frame #6: 0x00000001006fbd10 miniruby`vm_pop_frame(ec=0x00000001048038d0, cfp=0x0000000104a04440, ep=0x0000000104904a28) at vm_insnhelper.c:422:5
frame #7: 0x00000001006fbca0 miniruby`rb_vm_pop_frame(ec=0x00000001048038d0) at vm_insnhelper.c:431:5
frame #8: 0x00000001007d6420 miniruby`vm_call0_cfunc_with_frame(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:153:9
frame #9: 0x00000001007d44cc miniruby`vm_call0_cfunc(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:164:12
frame #10: 0x0000000100766e80 miniruby`vm_call0_body(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:210:15
frame #11: 0x00000001007d76f0 miniruby`vm_call0_cc(ec=0x00000001048038d0, recv=0x000000010d6b49d8, id=2769, argc=0, argv=0x0000000000000000, cc=0x000000010d6b2e58, kw_splat=0) at vm_eval.c:87:12
frame #12: 0x0000000100769e48 miniruby`rb_funcallv_scope(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000, scope=CALL_FCALL) at vm_eval.c:1051:16
frame #13: 0x0000000100760a54 miniruby`rb_funcallv(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000) at vm_eval.c:1066:12
frame #14: 0x000000010037513c miniruby`rb_inspect(obj=0x000000010d6b49d8) at object.c:633:34
frame #15: 0x000000010002c950 miniruby`inspect_ary(ary=0x000000010d6b4938, dummy=0x0000000000000000, recur=0) at array.c:3091:13
frame #16: 0x0000000100642020 miniruby`exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, pairid=0x0000000000000000, arg=0x0000000000000000, outer=0, mid=2769) at thread.c:5177:23
frame #17: 0x00000001006412fc miniruby`rb_exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, arg=0x0000000000000000) at thread.c:5205:12
frame #18: 0x00000001000127f0 miniruby`rb_ary_inspect(ary=0x000000010d6b4938) at array.c:3117:12
```
In general though, any calls back out to the interpreter could change
the IV buffer, so it's not safe to cache.
[Bug #19095]
2022-11-01 01:38:51 +03:00
struct iv_itr_data {
VALUE obj ;
struct gen_ivtbl * ivtbl ;
st_data_t arg ;
2022-12-09 01:16:52 +03:00
rb_ivar_foreach_callback_func * func ;
Always lookup IV buffers when iterating
Always look up instance variable buffers when iterating. It is possible
for the instance variable buffer to change out from under the object
during iteration, so we cannot cache the buffer on the stack.
In the case of Bug #19095, the transient heap moved the buffer during
iteration:
```
Watchpoint 1 hit:
old value: 0x0000000107c00df8
new value: 0x00000001032743c0
Process 31720 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
1358 }
1359 MEMCPY(new_ptr, old_ptr, VALUE, len);
1360 ROBJECT(obj)->as.heap.ivptr = new_ptr;
-> 1361 }
1362 }
1363 #endif
1364
miniruby`rb_obj_transient_heap_evacuate:
-> 0x1006e5178 <+328>: b 0x1006e517c ; <+332> at variable.c:1362:1
0x1006e517c <+332>: ldp x29, x30, [sp, #0x50]
0x1006e5180 <+336>: add sp, sp, #0x60
0x1006e5184 <+340>: ret
Target 0: (miniruby) stopped.
(lldb) bt
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
* frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
frame #1: 0x00000001006cb150 miniruby`transient_heap_block_evacuate(theap=0x0000000100b196c0, block=0x0000000107c00000) at transient_heap.c:734:17
frame #2: 0x00000001006c854c miniruby`transient_heap_evacuate(dmy=0x0000000000000000) at transient_heap.c:808:17
frame #3: 0x00000001007fe6c0 miniruby`rb_postponed_job_flush(vm=0x0000000104402900) at vm_trace.c:1773:21
frame #4: 0x0000000100637a84 miniruby`rb_threadptr_execute_interrupts(th=0x0000000103803bc0, blocking_timing=0) at thread.c:2316:13
frame #5: 0x000000010078b730 miniruby`rb_vm_check_ints(ec=0x00000001048038d0) at vm_core.h:2025:9
frame #6: 0x00000001006fbd10 miniruby`vm_pop_frame(ec=0x00000001048038d0, cfp=0x0000000104a04440, ep=0x0000000104904a28) at vm_insnhelper.c:422:5
frame #7: 0x00000001006fbca0 miniruby`rb_vm_pop_frame(ec=0x00000001048038d0) at vm_insnhelper.c:431:5
frame #8: 0x00000001007d6420 miniruby`vm_call0_cfunc_with_frame(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:153:9
frame #9: 0x00000001007d44cc miniruby`vm_call0_cfunc(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:164:12
frame #10: 0x0000000100766e80 miniruby`vm_call0_body(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:210:15
frame #11: 0x00000001007d76f0 miniruby`vm_call0_cc(ec=0x00000001048038d0, recv=0x000000010d6b49d8, id=2769, argc=0, argv=0x0000000000000000, cc=0x000000010d6b2e58, kw_splat=0) at vm_eval.c:87:12
frame #12: 0x0000000100769e48 miniruby`rb_funcallv_scope(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000, scope=CALL_FCALL) at vm_eval.c:1051:16
frame #13: 0x0000000100760a54 miniruby`rb_funcallv(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000) at vm_eval.c:1066:12
frame #14: 0x000000010037513c miniruby`rb_inspect(obj=0x000000010d6b49d8) at object.c:633:34
frame #15: 0x000000010002c950 miniruby`inspect_ary(ary=0x000000010d6b4938, dummy=0x0000000000000000, recur=0) at array.c:3091:13
frame #16: 0x0000000100642020 miniruby`exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, pairid=0x0000000000000000, arg=0x0000000000000000, outer=0, mid=2769) at thread.c:5177:23
frame #17: 0x00000001006412fc miniruby`rb_exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, arg=0x0000000000000000) at thread.c:5205:12
frame #18: 0x00000001000127f0 miniruby`rb_ary_inspect(ary=0x000000010d6b4938) at array.c:3117:12
```
In general though, any calls back out to the interpreter could change
the IV buffer, so it's not safe to cache.
[Bug #19095]
2022-11-01 01:38:51 +03:00
} ;
2023-06-20 06:39:22 +03:00
/*
* Returns a flag to stop iterating depending on the result of + callback + .
*/
static bool
Always lookup IV buffers when iterating
Always look up instance variable buffers when iterating. It is possible
for the instance variable buffer to change out from under the object
during iteration, so we cannot cache the buffer on the stack.
In the case of Bug #19095, the transient heap moved the buffer during
iteration:
```
Watchpoint 1 hit:
old value: 0x0000000107c00df8
new value: 0x00000001032743c0
Process 31720 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
1358 }
1359 MEMCPY(new_ptr, old_ptr, VALUE, len);
1360 ROBJECT(obj)->as.heap.ivptr = new_ptr;
-> 1361 }
1362 }
1363 #endif
1364
miniruby`rb_obj_transient_heap_evacuate:
-> 0x1006e5178 <+328>: b 0x1006e517c ; <+332> at variable.c:1362:1
0x1006e517c <+332>: ldp x29, x30, [sp, #0x50]
0x1006e5180 <+336>: add sp, sp, #0x60
0x1006e5184 <+340>: ret
Target 0: (miniruby) stopped.
(lldb) bt
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
* frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
frame #1: 0x00000001006cb150 miniruby`transient_heap_block_evacuate(theap=0x0000000100b196c0, block=0x0000000107c00000) at transient_heap.c:734:17
frame #2: 0x00000001006c854c miniruby`transient_heap_evacuate(dmy=0x0000000000000000) at transient_heap.c:808:17
frame #3: 0x00000001007fe6c0 miniruby`rb_postponed_job_flush(vm=0x0000000104402900) at vm_trace.c:1773:21
frame #4: 0x0000000100637a84 miniruby`rb_threadptr_execute_interrupts(th=0x0000000103803bc0, blocking_timing=0) at thread.c:2316:13
frame #5: 0x000000010078b730 miniruby`rb_vm_check_ints(ec=0x00000001048038d0) at vm_core.h:2025:9
frame #6: 0x00000001006fbd10 miniruby`vm_pop_frame(ec=0x00000001048038d0, cfp=0x0000000104a04440, ep=0x0000000104904a28) at vm_insnhelper.c:422:5
frame #7: 0x00000001006fbca0 miniruby`rb_vm_pop_frame(ec=0x00000001048038d0) at vm_insnhelper.c:431:5
frame #8: 0x00000001007d6420 miniruby`vm_call0_cfunc_with_frame(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:153:9
frame #9: 0x00000001007d44cc miniruby`vm_call0_cfunc(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:164:12
frame #10: 0x0000000100766e80 miniruby`vm_call0_body(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:210:15
frame #11: 0x00000001007d76f0 miniruby`vm_call0_cc(ec=0x00000001048038d0, recv=0x000000010d6b49d8, id=2769, argc=0, argv=0x0000000000000000, cc=0x000000010d6b2e58, kw_splat=0) at vm_eval.c:87:12
frame #12: 0x0000000100769e48 miniruby`rb_funcallv_scope(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000, scope=CALL_FCALL) at vm_eval.c:1051:16
frame #13: 0x0000000100760a54 miniruby`rb_funcallv(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000) at vm_eval.c:1066:12
frame #14: 0x000000010037513c miniruby`rb_inspect(obj=0x000000010d6b49d8) at object.c:633:34
frame #15: 0x000000010002c950 miniruby`inspect_ary(ary=0x000000010d6b4938, dummy=0x0000000000000000, recur=0) at array.c:3091:13
frame #16: 0x0000000100642020 miniruby`exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, pairid=0x0000000000000000, arg=0x0000000000000000, outer=0, mid=2769) at thread.c:5177:23
frame #17: 0x00000001006412fc miniruby`rb_exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, arg=0x0000000000000000) at thread.c:5205:12
frame #18: 0x00000001000127f0 miniruby`rb_ary_inspect(ary=0x000000010d6b4938) at array.c:3117:12
```
In general though, any calls back out to the interpreter could change
the IV buffer, so it's not safe to cache.
[Bug #19095]
2022-11-01 01:38:51 +03:00
iterate_over_shapes_with_callback ( rb_shape_t * shape , rb_ivar_foreach_callback_func * callback , struct iv_itr_data * itr_data )
2022-10-24 11:42:29 +03:00
{
2022-10-03 18:14:32 +03:00
switch ( ( enum shape_type ) shape - > type ) {
2022-10-24 11:42:29 +03:00
case SHAPE_ROOT :
2024-03-12 21:27:34 +03:00
case SHAPE_T_OBJECT :
2023-06-20 06:39:22 +03:00
return false ;
2022-10-24 11:42:29 +03:00
case SHAPE_IVAR :
2023-06-20 06:39:22 +03:00
ASSUME ( callback ) ;
if ( iterate_over_shapes_with_callback ( rb_shape_get_parent ( shape ) , callback , itr_data ) )
return true ;
Always lookup IV buffers when iterating
Always look up instance variable buffers when iterating. It is possible
for the instance variable buffer to change out from under the object
during iteration, so we cannot cache the buffer on the stack.
In the case of Bug #19095, the transient heap moved the buffer during
iteration:
```
Watchpoint 1 hit:
old value: 0x0000000107c00df8
new value: 0x00000001032743c0
Process 31720 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
1358 }
1359 MEMCPY(new_ptr, old_ptr, VALUE, len);
1360 ROBJECT(obj)->as.heap.ivptr = new_ptr;
-> 1361 }
1362 }
1363 #endif
1364
miniruby`rb_obj_transient_heap_evacuate:
-> 0x1006e5178 <+328>: b 0x1006e517c ; <+332> at variable.c:1362:1
0x1006e517c <+332>: ldp x29, x30, [sp, #0x50]
0x1006e5180 <+336>: add sp, sp, #0x60
0x1006e5184 <+340>: ret
Target 0: (miniruby) stopped.
(lldb) bt
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
* frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
frame #1: 0x00000001006cb150 miniruby`transient_heap_block_evacuate(theap=0x0000000100b196c0, block=0x0000000107c00000) at transient_heap.c:734:17
frame #2: 0x00000001006c854c miniruby`transient_heap_evacuate(dmy=0x0000000000000000) at transient_heap.c:808:17
frame #3: 0x00000001007fe6c0 miniruby`rb_postponed_job_flush(vm=0x0000000104402900) at vm_trace.c:1773:21
frame #4: 0x0000000100637a84 miniruby`rb_threadptr_execute_interrupts(th=0x0000000103803bc0, blocking_timing=0) at thread.c:2316:13
frame #5: 0x000000010078b730 miniruby`rb_vm_check_ints(ec=0x00000001048038d0) at vm_core.h:2025:9
frame #6: 0x00000001006fbd10 miniruby`vm_pop_frame(ec=0x00000001048038d0, cfp=0x0000000104a04440, ep=0x0000000104904a28) at vm_insnhelper.c:422:5
frame #7: 0x00000001006fbca0 miniruby`rb_vm_pop_frame(ec=0x00000001048038d0) at vm_insnhelper.c:431:5
frame #8: 0x00000001007d6420 miniruby`vm_call0_cfunc_with_frame(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:153:9
frame #9: 0x00000001007d44cc miniruby`vm_call0_cfunc(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:164:12
frame #10: 0x0000000100766e80 miniruby`vm_call0_body(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:210:15
frame #11: 0x00000001007d76f0 miniruby`vm_call0_cc(ec=0x00000001048038d0, recv=0x000000010d6b49d8, id=2769, argc=0, argv=0x0000000000000000, cc=0x000000010d6b2e58, kw_splat=0) at vm_eval.c:87:12
frame #12: 0x0000000100769e48 miniruby`rb_funcallv_scope(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000, scope=CALL_FCALL) at vm_eval.c:1051:16
frame #13: 0x0000000100760a54 miniruby`rb_funcallv(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000) at vm_eval.c:1066:12
frame #14: 0x000000010037513c miniruby`rb_inspect(obj=0x000000010d6b49d8) at object.c:633:34
frame #15: 0x000000010002c950 miniruby`inspect_ary(ary=0x000000010d6b4938, dummy=0x0000000000000000, recur=0) at array.c:3091:13
frame #16: 0x0000000100642020 miniruby`exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, pairid=0x0000000000000000, arg=0x0000000000000000, outer=0, mid=2769) at thread.c:5177:23
frame #17: 0x00000001006412fc miniruby`rb_exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, arg=0x0000000000000000) at thread.c:5205:12
frame #18: 0x00000001000127f0 miniruby`rb_ary_inspect(ary=0x000000010d6b4938) at array.c:3117:12
```
In general though, any calls back out to the interpreter could change
the IV buffer, so it's not safe to cache.
[Bug #19095]
2022-11-01 01:38:51 +03:00
VALUE * iv_list ;
2022-11-01 03:35:09 +03:00
switch ( BUILTIN_TYPE ( itr_data - > obj ) ) {
case T_OBJECT :
2022-12-09 01:16:52 +03:00
RUBY_ASSERT ( ! rb_shape_obj_too_complex ( itr_data - > obj ) ) ;
2022-11-01 03:35:09 +03:00
iv_list = ROBJECT_IVPTR ( itr_data - > obj ) ;
break ;
case T_CLASS :
case T_MODULE :
iv_list = RCLASS_IVPTR ( itr_data - > obj ) ;
break ;
default :
2023-10-31 16:38:35 +03:00
iv_list = itr_data - > ivtbl - > as . shape . ivptr ;
2022-11-01 03:35:09 +03:00
break ;
Always lookup IV buffers when iterating
Always look up instance variable buffers when iterating. It is possible
for the instance variable buffer to change out from under the object
during iteration, so we cannot cache the buffer on the stack.
In the case of Bug #19095, the transient heap moved the buffer during
iteration:
```
Watchpoint 1 hit:
old value: 0x0000000107c00df8
new value: 0x00000001032743c0
Process 31720 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
1358 }
1359 MEMCPY(new_ptr, old_ptr, VALUE, len);
1360 ROBJECT(obj)->as.heap.ivptr = new_ptr;
-> 1361 }
1362 }
1363 #endif
1364
miniruby`rb_obj_transient_heap_evacuate:
-> 0x1006e5178 <+328>: b 0x1006e517c ; <+332> at variable.c:1362:1
0x1006e517c <+332>: ldp x29, x30, [sp, #0x50]
0x1006e5180 <+336>: add sp, sp, #0x60
0x1006e5184 <+340>: ret
Target 0: (miniruby) stopped.
(lldb) bt
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
* frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
frame #1: 0x00000001006cb150 miniruby`transient_heap_block_evacuate(theap=0x0000000100b196c0, block=0x0000000107c00000) at transient_heap.c:734:17
frame #2: 0x00000001006c854c miniruby`transient_heap_evacuate(dmy=0x0000000000000000) at transient_heap.c:808:17
frame #3: 0x00000001007fe6c0 miniruby`rb_postponed_job_flush(vm=0x0000000104402900) at vm_trace.c:1773:21
frame #4: 0x0000000100637a84 miniruby`rb_threadptr_execute_interrupts(th=0x0000000103803bc0, blocking_timing=0) at thread.c:2316:13
frame #5: 0x000000010078b730 miniruby`rb_vm_check_ints(ec=0x00000001048038d0) at vm_core.h:2025:9
frame #6: 0x00000001006fbd10 miniruby`vm_pop_frame(ec=0x00000001048038d0, cfp=0x0000000104a04440, ep=0x0000000104904a28) at vm_insnhelper.c:422:5
frame #7: 0x00000001006fbca0 miniruby`rb_vm_pop_frame(ec=0x00000001048038d0) at vm_insnhelper.c:431:5
frame #8: 0x00000001007d6420 miniruby`vm_call0_cfunc_with_frame(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:153:9
frame #9: 0x00000001007d44cc miniruby`vm_call0_cfunc(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:164:12
frame #10: 0x0000000100766e80 miniruby`vm_call0_body(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:210:15
frame #11: 0x00000001007d76f0 miniruby`vm_call0_cc(ec=0x00000001048038d0, recv=0x000000010d6b49d8, id=2769, argc=0, argv=0x0000000000000000, cc=0x000000010d6b2e58, kw_splat=0) at vm_eval.c:87:12
frame #12: 0x0000000100769e48 miniruby`rb_funcallv_scope(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000, scope=CALL_FCALL) at vm_eval.c:1051:16
frame #13: 0x0000000100760a54 miniruby`rb_funcallv(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000) at vm_eval.c:1066:12
frame #14: 0x000000010037513c miniruby`rb_inspect(obj=0x000000010d6b49d8) at object.c:633:34
frame #15: 0x000000010002c950 miniruby`inspect_ary(ary=0x000000010d6b4938, dummy=0x0000000000000000, recur=0) at array.c:3091:13
frame #16: 0x0000000100642020 miniruby`exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, pairid=0x0000000000000000, arg=0x0000000000000000, outer=0, mid=2769) at thread.c:5177:23
frame #17: 0x00000001006412fc miniruby`rb_exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, arg=0x0000000000000000) at thread.c:5205:12
frame #18: 0x00000001000127f0 miniruby`rb_ary_inspect(ary=0x000000010d6b4938) at array.c:3117:12
```
In general though, any calls back out to the interpreter could change
the IV buffer, so it's not safe to cache.
[Bug #19095]
2022-11-01 01:38:51 +03:00
}
2022-10-24 11:42:29 +03:00
VALUE val = iv_list [ shape - > next_iv_index - 1 ] ;
2022-11-15 07:24:08 +03:00
if ( ! UNDEF_P ( val ) ) {
2023-06-20 06:39:22 +03:00
switch ( callback ( shape - > edge_name , val , itr_data - > arg ) ) {
case ST_CHECK :
case ST_CONTINUE :
break ;
case ST_STOP :
return true ;
default :
rb_bug ( " unreachable " ) ;
}
2022-10-24 11:42:29 +03:00
}
2023-06-20 06:39:22 +03:00
return false ;
2022-10-24 11:42:29 +03:00
case SHAPE_FROZEN :
2023-06-20 06:39:22 +03:00
return iterate_over_shapes_with_callback ( rb_shape_get_parent ( shape ) , callback , itr_data ) ;
2022-12-09 01:16:52 +03:00
case SHAPE_OBJ_TOO_COMPLEX :
2023-06-22 00:31:20 +03:00
default :
2023-05-20 08:00:14 +03:00
rb_bug ( " Unreachable " ) ;
2007-09-28 10:21:46 +04:00
}
}
2023-03-17 21:29:04 +03:00
static int
each_hash_iv ( st_data_t id , st_data_t val , st_data_t data )
2022-12-09 01:16:52 +03:00
{
struct iv_itr_data * itr_data = ( struct iv_itr_data * ) data ;
rb_ivar_foreach_callback_func * callback = itr_data - > func ;
2023-03-17 21:29:04 +03:00
return callback ( ( ID ) id , ( VALUE ) val , itr_data - > arg ) ;
2022-12-09 01:16:52 +03:00
}
2007-09-28 10:21:46 +04:00
static void
2019-08-27 06:08:32 +03:00
obj_ivar_each ( VALUE obj , rb_ivar_foreach_callback_func * func , st_data_t arg )
2007-09-28 10:21:46 +04:00
{
2022-10-03 18:14:32 +03:00
rb_shape_t * shape = rb_shape_get_shape ( obj ) ;
Always lookup IV buffers when iterating
Always look up instance variable buffers when iterating. It is possible
for the instance variable buffer to change out from under the object
during iteration, so we cannot cache the buffer on the stack.
In the case of Bug #19095, the transient heap moved the buffer during
iteration:
```
Watchpoint 1 hit:
old value: 0x0000000107c00df8
new value: 0x00000001032743c0
Process 31720 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
1358 }
1359 MEMCPY(new_ptr, old_ptr, VALUE, len);
1360 ROBJECT(obj)->as.heap.ivptr = new_ptr;
-> 1361 }
1362 }
1363 #endif
1364
miniruby`rb_obj_transient_heap_evacuate:
-> 0x1006e5178 <+328>: b 0x1006e517c ; <+332> at variable.c:1362:1
0x1006e517c <+332>: ldp x29, x30, [sp, #0x50]
0x1006e5180 <+336>: add sp, sp, #0x60
0x1006e5184 <+340>: ret
Target 0: (miniruby) stopped.
(lldb) bt
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
* frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
frame #1: 0x00000001006cb150 miniruby`transient_heap_block_evacuate(theap=0x0000000100b196c0, block=0x0000000107c00000) at transient_heap.c:734:17
frame #2: 0x00000001006c854c miniruby`transient_heap_evacuate(dmy=0x0000000000000000) at transient_heap.c:808:17
frame #3: 0x00000001007fe6c0 miniruby`rb_postponed_job_flush(vm=0x0000000104402900) at vm_trace.c:1773:21
frame #4: 0x0000000100637a84 miniruby`rb_threadptr_execute_interrupts(th=0x0000000103803bc0, blocking_timing=0) at thread.c:2316:13
frame #5: 0x000000010078b730 miniruby`rb_vm_check_ints(ec=0x00000001048038d0) at vm_core.h:2025:9
frame #6: 0x00000001006fbd10 miniruby`vm_pop_frame(ec=0x00000001048038d0, cfp=0x0000000104a04440, ep=0x0000000104904a28) at vm_insnhelper.c:422:5
frame #7: 0x00000001006fbca0 miniruby`rb_vm_pop_frame(ec=0x00000001048038d0) at vm_insnhelper.c:431:5
frame #8: 0x00000001007d6420 miniruby`vm_call0_cfunc_with_frame(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:153:9
frame #9: 0x00000001007d44cc miniruby`vm_call0_cfunc(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:164:12
frame #10: 0x0000000100766e80 miniruby`vm_call0_body(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:210:15
frame #11: 0x00000001007d76f0 miniruby`vm_call0_cc(ec=0x00000001048038d0, recv=0x000000010d6b49d8, id=2769, argc=0, argv=0x0000000000000000, cc=0x000000010d6b2e58, kw_splat=0) at vm_eval.c:87:12
frame #12: 0x0000000100769e48 miniruby`rb_funcallv_scope(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000, scope=CALL_FCALL) at vm_eval.c:1051:16
frame #13: 0x0000000100760a54 miniruby`rb_funcallv(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000) at vm_eval.c:1066:12
frame #14: 0x000000010037513c miniruby`rb_inspect(obj=0x000000010d6b49d8) at object.c:633:34
frame #15: 0x000000010002c950 miniruby`inspect_ary(ary=0x000000010d6b4938, dummy=0x0000000000000000, recur=0) at array.c:3091:13
frame #16: 0x0000000100642020 miniruby`exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, pairid=0x0000000000000000, arg=0x0000000000000000, outer=0, mid=2769) at thread.c:5177:23
frame #17: 0x00000001006412fc miniruby`rb_exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, arg=0x0000000000000000) at thread.c:5205:12
frame #18: 0x00000001000127f0 miniruby`rb_ary_inspect(ary=0x000000010d6b4938) at array.c:3117:12
```
In general though, any calls back out to the interpreter could change
the IV buffer, so it's not safe to cache.
[Bug #19095]
2022-11-01 01:38:51 +03:00
struct iv_itr_data itr_data ;
itr_data . obj = obj ;
itr_data . arg = arg ;
2022-12-09 01:16:52 +03:00
itr_data . func = func ;
if ( rb_shape_obj_too_complex ( obj ) ) {
2023-03-17 21:29:04 +03:00
rb_st_foreach ( ROBJECT_IV_HASH ( obj ) , each_hash_iv , ( st_data_t ) & itr_data ) ;
2022-12-09 01:16:52 +03:00
}
else {
iterate_over_shapes_with_callback ( shape , func , & itr_data ) ;
}
2015-05-30 02:42:49 +03:00
}
static void
2019-08-27 06:08:32 +03:00
gen_ivar_each ( VALUE obj , rb_ivar_foreach_callback_func * func , st_data_t arg )
2015-05-30 02:42:49 +03:00
{
2022-10-03 18:14:32 +03:00
rb_shape_t * shape = rb_shape_get_shape ( obj ) ;
2020-10-16 09:20:40 +03:00
struct gen_ivtbl * ivtbl ;
2022-10-03 18:14:32 +03:00
if ( ! rb_gen_ivtbl_get ( obj , 0 , & ivtbl ) ) return ;
2022-09-27 02:09:50 +03:00
Always lookup IV buffers when iterating
Always look up instance variable buffers when iterating. It is possible
for the instance variable buffer to change out from under the object
during iteration, so we cannot cache the buffer on the stack.
In the case of Bug #19095, the transient heap moved the buffer during
iteration:
```
Watchpoint 1 hit:
old value: 0x0000000107c00df8
new value: 0x00000001032743c0
Process 31720 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
1358 }
1359 MEMCPY(new_ptr, old_ptr, VALUE, len);
1360 ROBJECT(obj)->as.heap.ivptr = new_ptr;
-> 1361 }
1362 }
1363 #endif
1364
miniruby`rb_obj_transient_heap_evacuate:
-> 0x1006e5178 <+328>: b 0x1006e517c ; <+332> at variable.c:1362:1
0x1006e517c <+332>: ldp x29, x30, [sp, #0x50]
0x1006e5180 <+336>: add sp, sp, #0x60
0x1006e5184 <+340>: ret
Target 0: (miniruby) stopped.
(lldb) bt
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
* frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
frame #1: 0x00000001006cb150 miniruby`transient_heap_block_evacuate(theap=0x0000000100b196c0, block=0x0000000107c00000) at transient_heap.c:734:17
frame #2: 0x00000001006c854c miniruby`transient_heap_evacuate(dmy=0x0000000000000000) at transient_heap.c:808:17
frame #3: 0x00000001007fe6c0 miniruby`rb_postponed_job_flush(vm=0x0000000104402900) at vm_trace.c:1773:21
frame #4: 0x0000000100637a84 miniruby`rb_threadptr_execute_interrupts(th=0x0000000103803bc0, blocking_timing=0) at thread.c:2316:13
frame #5: 0x000000010078b730 miniruby`rb_vm_check_ints(ec=0x00000001048038d0) at vm_core.h:2025:9
frame #6: 0x00000001006fbd10 miniruby`vm_pop_frame(ec=0x00000001048038d0, cfp=0x0000000104a04440, ep=0x0000000104904a28) at vm_insnhelper.c:422:5
frame #7: 0x00000001006fbca0 miniruby`rb_vm_pop_frame(ec=0x00000001048038d0) at vm_insnhelper.c:431:5
frame #8: 0x00000001007d6420 miniruby`vm_call0_cfunc_with_frame(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:153:9
frame #9: 0x00000001007d44cc miniruby`vm_call0_cfunc(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:164:12
frame #10: 0x0000000100766e80 miniruby`vm_call0_body(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:210:15
frame #11: 0x00000001007d76f0 miniruby`vm_call0_cc(ec=0x00000001048038d0, recv=0x000000010d6b49d8, id=2769, argc=0, argv=0x0000000000000000, cc=0x000000010d6b2e58, kw_splat=0) at vm_eval.c:87:12
frame #12: 0x0000000100769e48 miniruby`rb_funcallv_scope(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000, scope=CALL_FCALL) at vm_eval.c:1051:16
frame #13: 0x0000000100760a54 miniruby`rb_funcallv(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000) at vm_eval.c:1066:12
frame #14: 0x000000010037513c miniruby`rb_inspect(obj=0x000000010d6b49d8) at object.c:633:34
frame #15: 0x000000010002c950 miniruby`inspect_ary(ary=0x000000010d6b4938, dummy=0x0000000000000000, recur=0) at array.c:3091:13
frame #16: 0x0000000100642020 miniruby`exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, pairid=0x0000000000000000, arg=0x0000000000000000, outer=0, mid=2769) at thread.c:5177:23
frame #17: 0x00000001006412fc miniruby`rb_exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, arg=0x0000000000000000) at thread.c:5205:12
frame #18: 0x00000001000127f0 miniruby`rb_ary_inspect(ary=0x000000010d6b4938) at array.c:3117:12
```
In general though, any calls back out to the interpreter could change
the IV buffer, so it's not safe to cache.
[Bug #19095]
2022-11-01 01:38:51 +03:00
struct iv_itr_data itr_data ;
itr_data . obj = obj ;
itr_data . ivtbl = ivtbl ;
itr_data . arg = arg ;
2023-10-20 02:01:35 +03:00
itr_data . func = func ;
if ( rb_shape_obj_too_complex ( obj ) ) {
2023-11-20 18:13:18 +03:00
rb_st_foreach ( ivtbl - > as . complex . table , each_hash_iv , ( st_data_t ) & itr_data ) ;
2023-10-20 02:01:35 +03:00
}
else {
iterate_over_shapes_with_callback ( shape , func , & itr_data ) ;
}
2015-05-30 02:42:49 +03:00
}
2022-11-01 00:05:37 +03:00
static void
class_ivar_each ( VALUE obj , rb_ivar_foreach_callback_func * func , st_data_t arg )
{
RUBY_ASSERT ( RB_TYPE_P ( obj , T_CLASS ) | | RB_TYPE_P ( obj , T_MODULE ) ) ;
rb_shape_t * shape = rb_shape_get_shape ( obj ) ;
Always lookup IV buffers when iterating
Always look up instance variable buffers when iterating. It is possible
for the instance variable buffer to change out from under the object
during iteration, so we cannot cache the buffer on the stack.
In the case of Bug #19095, the transient heap moved the buffer during
iteration:
```
Watchpoint 1 hit:
old value: 0x0000000107c00df8
new value: 0x00000001032743c0
Process 31720 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
1358 }
1359 MEMCPY(new_ptr, old_ptr, VALUE, len);
1360 ROBJECT(obj)->as.heap.ivptr = new_ptr;
-> 1361 }
1362 }
1363 #endif
1364
miniruby`rb_obj_transient_heap_evacuate:
-> 0x1006e5178 <+328>: b 0x1006e517c ; <+332> at variable.c:1362:1
0x1006e517c <+332>: ldp x29, x30, [sp, #0x50]
0x1006e5180 <+336>: add sp, sp, #0x60
0x1006e5184 <+340>: ret
Target 0: (miniruby) stopped.
(lldb) bt
* thread #1, queue = 'com.apple.main-thread', stop reason = watchpoint 1
* frame #0: 0x00000001006e5178 miniruby`rb_obj_transient_heap_evacuate(obj=0x000000010d6b94b0, promote=1) at variable.c:1361:5
frame #1: 0x00000001006cb150 miniruby`transient_heap_block_evacuate(theap=0x0000000100b196c0, block=0x0000000107c00000) at transient_heap.c:734:17
frame #2: 0x00000001006c854c miniruby`transient_heap_evacuate(dmy=0x0000000000000000) at transient_heap.c:808:17
frame #3: 0x00000001007fe6c0 miniruby`rb_postponed_job_flush(vm=0x0000000104402900) at vm_trace.c:1773:21
frame #4: 0x0000000100637a84 miniruby`rb_threadptr_execute_interrupts(th=0x0000000103803bc0, blocking_timing=0) at thread.c:2316:13
frame #5: 0x000000010078b730 miniruby`rb_vm_check_ints(ec=0x00000001048038d0) at vm_core.h:2025:9
frame #6: 0x00000001006fbd10 miniruby`vm_pop_frame(ec=0x00000001048038d0, cfp=0x0000000104a04440, ep=0x0000000104904a28) at vm_insnhelper.c:422:5
frame #7: 0x00000001006fbca0 miniruby`rb_vm_pop_frame(ec=0x00000001048038d0) at vm_insnhelper.c:431:5
frame #8: 0x00000001007d6420 miniruby`vm_call0_cfunc_with_frame(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:153:9
frame #9: 0x00000001007d44cc miniruby`vm_call0_cfunc(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:164:12
frame #10: 0x0000000100766e80 miniruby`vm_call0_body(ec=0x00000001048038d0, calling=0x000000016fdcc6a0, argv=0x0000000000000000) at vm_eval.c:210:15
frame #11: 0x00000001007d76f0 miniruby`vm_call0_cc(ec=0x00000001048038d0, recv=0x000000010d6b49d8, id=2769, argc=0, argv=0x0000000000000000, cc=0x000000010d6b2e58, kw_splat=0) at vm_eval.c:87:12
frame #12: 0x0000000100769e48 miniruby`rb_funcallv_scope(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000, scope=CALL_FCALL) at vm_eval.c:1051:16
frame #13: 0x0000000100760a54 miniruby`rb_funcallv(recv=0x000000010d6b49d8, mid=2769, argc=0, argv=0x0000000000000000) at vm_eval.c:1066:12
frame #14: 0x000000010037513c miniruby`rb_inspect(obj=0x000000010d6b49d8) at object.c:633:34
frame #15: 0x000000010002c950 miniruby`inspect_ary(ary=0x000000010d6b4938, dummy=0x0000000000000000, recur=0) at array.c:3091:13
frame #16: 0x0000000100642020 miniruby`exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, pairid=0x0000000000000000, arg=0x0000000000000000, outer=0, mid=2769) at thread.c:5177:23
frame #17: 0x00000001006412fc miniruby`rb_exec_recursive(func=(miniruby`inspect_ary at array.c:3084), obj=0x000000010d6b4938, arg=0x0000000000000000) at thread.c:5205:12
frame #18: 0x00000001000127f0 miniruby`rb_ary_inspect(ary=0x000000010d6b4938) at array.c:3117:12
```
In general though, any calls back out to the interpreter could change
the IV buffer, so it's not safe to cache.
[Bug #19095]
2022-11-01 01:38:51 +03:00
struct iv_itr_data itr_data ;
itr_data . obj = obj ;
itr_data . arg = arg ;
2023-10-19 21:00:54 +03:00
itr_data . func = func ;
if ( rb_shape_obj_too_complex ( obj ) ) {
rb_st_foreach ( RCLASS_IV_HASH ( obj ) , each_hash_iv , ( st_data_t ) & itr_data ) ;
}
else {
iterate_over_shapes_with_callback ( shape , func , & itr_data ) ;
}
2022-11-01 00:05:37 +03:00
}
2015-05-30 02:42:49 +03:00
void
rb_copy_generic_ivar ( VALUE clone , VALUE obj )
{
2022-10-03 18:14:32 +03:00
struct gen_ivtbl * obj_ivtbl ;
struct gen_ivtbl * new_ivtbl ;
2015-05-30 02:42:49 +03:00
variable.c: remove generic ivar support for special constants
Special constants are all frozen since [Feature #8923] and cannot
support ivars. Remove some unused code we had for supporting them.
* variable.c (special_generic_ivar): remove flag
(givar_i, rb_mark_generic_ivar_tbl): remove functions
(rb_free_generic_ivar, rb_ivar_lookup, rb_ivar_delete,
generic_ivar_set, rb_ivar_set, rb_ivar_defined,
rb_copy_generic_ivar, rb_ivar_foreach, rb_ivar_count,
rb_obj_remove_instance_variable):
adjust for lack of ivar support in special constants
* test/ruby/test_variable.rb: test ivars for special consts
* internal.h: remove rb_mark_generic_ivar_tbl decl
* gc.c (gc_mark_roots): remove rb_mark_generic_ivar_tbl call
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@50758 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-06-03 23:53:35 +03:00
rb_check_frozen ( clone ) ;
2015-05-30 02:42:49 +03:00
if ( ! FL_TEST ( obj , FL_EXIVAR ) ) {
2020-06-22 04:15:32 +03:00
goto clear ;
2015-05-30 02:42:49 +03:00
}
2022-10-03 18:14:32 +03:00
if ( rb_gen_ivtbl_get ( obj , 0 , & obj_ivtbl ) ) {
2023-10-31 16:38:35 +03:00
if ( gen_ivtbl_count ( obj , obj_ivtbl ) = = 0 )
2015-05-30 02:42:49 +03:00
goto clear ;
2022-10-03 18:14:32 +03:00
FL_SET ( clone , FL_EXIVAR ) ;
2023-10-31 16:38:35 +03:00
if ( rb_shape_obj_too_complex ( obj ) ) {
new_ivtbl = xmalloc ( sizeof ( struct gen_ivtbl ) ) ;
# if !SHAPE_IN_BASIC_FLAGS
new_ivtbl - > shape_id = SHAPE_OBJ_TOO_COMPLEX ;
# endif
new_ivtbl - > as . complex . table = st_copy ( obj_ivtbl - > as . complex . table ) ;
}
else {
new_ivtbl = gen_ivtbl_resize ( 0 , obj_ivtbl - > as . shape . numiv ) ;
for ( uint32_t i = 0 ; i < obj_ivtbl - > as . shape . numiv ; i + + ) {
2023-11-03 18:00:58 +03:00
RB_OBJ_WRITE ( clone , & new_ivtbl - > as . shape . ivptr [ i ] , obj_ivtbl - > as . shape . ivptr [ i ] ) ;
2023-10-31 16:38:35 +03:00
}
2015-05-30 02:42:49 +03:00
}
/*
* c . ivtbl may change in gen_ivar_copy due to realloc ,
* no need to free
*/
2020-10-14 04:43:13 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
generic_ivtbl_no_ractor_check ( clone ) ;
2022-10-03 18:14:32 +03:00
st_insert ( generic_ivtbl_no_ractor_check ( obj ) , ( st_data_t ) clone , ( st_data_t ) new_ivtbl ) ;
2020-10-14 04:43:13 +03:00
}
RB_VM_LOCK_LEAVE ( ) ;
2022-10-03 18:14:32 +03:00
rb_shape_t * obj_shape = rb_shape_get_shape ( obj ) ;
if ( rb_shape_frozen_shape_p ( obj_shape ) ) {
2022-10-03 20:52:40 +03:00
rb_shape_set_shape_id ( clone , obj_shape - > parent_id ) ;
2022-10-03 18:14:32 +03:00
}
else {
rb_shape_set_shape ( clone , obj_shape ) ;
}
2015-05-30 02:42:49 +03:00
}
2020-06-22 04:15:32 +03:00
return ;
clear :
if ( FL_TEST ( clone , FL_EXIVAR ) ) {
rb_free_generic_ivar ( clone ) ;
FL_UNSET ( clone , FL_EXIVAR ) ;
}
2015-05-30 02:42:49 +03:00
}
2020-11-01 03:56:40 +03:00
void
rb_replace_generic_ivar ( VALUE clone , VALUE obj )
{
RUBY_ASSERT ( FL_TEST ( obj , FL_EXIVAR ) ) ;
RB_VM_LOCK_ENTER ( ) ;
{
2021-06-23 04:50:09 +03:00
st_data_t ivtbl , obj_data = ( st_data_t ) obj ;
if ( st_lookup ( generic_iv_tbl_ , ( st_data_t ) obj , & ivtbl ) ) {
st_insert ( generic_iv_tbl_ , ( st_data_t ) clone , ivtbl ) ;
st_delete ( generic_iv_tbl_ , & obj_data , NULL ) ;
2020-11-01 03:56:40 +03:00
}
else {
rb_bug ( " unreachable " ) ;
}
}
RB_VM_LOCK_LEAVE ( ) ;
FL_SET ( clone , FL_EXIVAR ) ;
}
2009-12-05 08:45:29 +03:00
void
2019-08-27 06:08:32 +03:00
rb_ivar_foreach ( VALUE obj , rb_ivar_foreach_callback_func * func , st_data_t arg )
2007-09-28 10:21:46 +04:00
{
variable.c: remove generic ivar support for special constants
Special constants are all frozen since [Feature #8923] and cannot
support ivars. Remove some unused code we had for supporting them.
* variable.c (special_generic_ivar): remove flag
(givar_i, rb_mark_generic_ivar_tbl): remove functions
(rb_free_generic_ivar, rb_ivar_lookup, rb_ivar_delete,
generic_ivar_set, rb_ivar_set, rb_ivar_defined,
rb_copy_generic_ivar, rb_ivar_foreach, rb_ivar_count,
rb_obj_remove_instance_variable):
adjust for lack of ivar support in special constants
* test/ruby/test_variable.rb: test ivars for special consts
* internal.h: remove rb_mark_generic_ivar_tbl decl
* gc.c (gc_mark_roots): remove rb_mark_generic_ivar_tbl call
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@50758 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-06-03 23:53:35 +03:00
if ( SPECIAL_CONST_P ( obj ) ) return ;
2012-07-28 21:04:58 +04:00
switch ( BUILTIN_TYPE ( obj ) ) {
2007-09-28 10:21:46 +04:00
case T_OBJECT :
obj_ivar_each ( obj , func , arg ) ;
break ;
case T_CLASS :
case T_MODULE :
2020-03-09 20:22:11 +03:00
IVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR ( 0 ) ;
2022-11-01 00:05:37 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
class_ivar_each ( obj , func , arg ) ;
2007-09-28 10:21:46 +04:00
}
2022-11-01 00:05:37 +03:00
RB_VM_LOCK_LEAVE ( ) ;
2007-09-28 10:21:46 +04:00
break ;
default :
variable.c: remove generic ivar support for special constants
Special constants are all frozen since [Feature #8923] and cannot
support ivars. Remove some unused code we had for supporting them.
* variable.c (special_generic_ivar): remove flag
(givar_i, rb_mark_generic_ivar_tbl): remove functions
(rb_free_generic_ivar, rb_ivar_lookup, rb_ivar_delete,
generic_ivar_set, rb_ivar_set, rb_ivar_defined,
rb_copy_generic_ivar, rb_ivar_foreach, rb_ivar_count,
rb_obj_remove_instance_variable):
adjust for lack of ivar support in special constants
* test/ruby/test_variable.rb: test ivars for special consts
* internal.h: remove rb_mark_generic_ivar_tbl decl
* gc.c (gc_mark_roots): remove rb_mark_generic_ivar_tbl call
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@50758 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-06-03 23:53:35 +03:00
if ( FL_TEST ( obj , FL_EXIVAR ) ) {
2015-05-30 02:42:49 +03:00
gen_ivar_each ( obj , func , arg ) ;
2007-09-28 10:21:46 +04:00
}
break ;
}
}
2009-12-05 09:35:07 +03:00
st_index_t
rb_ivar_count ( VALUE obj )
{
variable.c: remove generic ivar support for special constants
Special constants are all frozen since [Feature #8923] and cannot
support ivars. Remove some unused code we had for supporting them.
* variable.c (special_generic_ivar): remove flag
(givar_i, rb_mark_generic_ivar_tbl): remove functions
(rb_free_generic_ivar, rb_ivar_lookup, rb_ivar_delete,
generic_ivar_set, rb_ivar_set, rb_ivar_defined,
rb_copy_generic_ivar, rb_ivar_foreach, rb_ivar_count,
rb_obj_remove_instance_variable):
adjust for lack of ivar support in special constants
* test/ruby/test_variable.rb: test ivars for special consts
* internal.h: remove rb_mark_generic_ivar_tbl decl
* gc.c (gc_mark_roots): remove rb_mark_generic_ivar_tbl call
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@50758 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-06-03 23:53:35 +03:00
if ( SPECIAL_CONST_P ( obj ) ) return 0 ;
2012-07-28 21:04:58 +04:00
switch ( BUILTIN_TYPE ( obj ) ) {
2009-12-05 09:35:07 +03:00
case T_OBJECT :
2023-11-21 20:23:37 +03:00
return ROBJECT_IV_COUNT ( obj ) ;
2009-12-05 09:35:07 +03:00
case T_CLASS :
case T_MODULE :
2023-11-21 20:23:37 +03:00
return RCLASS_IV_COUNT ( obj ) ;
2009-12-05 09:35:07 +03:00
default :
variable.c: remove generic ivar support for special constants
Special constants are all frozen since [Feature #8923] and cannot
support ivars. Remove some unused code we had for supporting them.
* variable.c (special_generic_ivar): remove flag
(givar_i, rb_mark_generic_ivar_tbl): remove functions
(rb_free_generic_ivar, rb_ivar_lookup, rb_ivar_delete,
generic_ivar_set, rb_ivar_set, rb_ivar_defined,
rb_copy_generic_ivar, rb_ivar_foreach, rb_ivar_count,
rb_obj_remove_instance_variable):
adjust for lack of ivar support in special constants
* test/ruby/test_variable.rb: test ivars for special consts
* internal.h: remove rb_mark_generic_ivar_tbl decl
* gc.c (gc_mark_roots): remove rb_mark_generic_ivar_tbl call
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@50758 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-06-03 23:53:35 +03:00
if ( FL_TEST ( obj , FL_EXIVAR ) ) {
2015-05-30 02:42:49 +03:00
struct gen_ivtbl * ivtbl ;
2009-12-05 09:35:07 +03:00
2022-10-03 18:14:32 +03:00
if ( rb_gen_ivtbl_get ( obj , 0 , & ivtbl ) ) {
2023-10-31 16:38:35 +03:00
return gen_ivtbl_count ( obj , ivtbl ) ;
2022-09-28 18:27:12 +03:00
}
}
break ;
2009-12-05 09:35:07 +03:00
}
return 0 ;
}
2007-09-28 10:21:46 +04:00
static int
2023-12-06 01:54:37 +03:00
ivar_i ( ID key , VALUE v , st_data_t a )
1998-01-16 15:19:22 +03:00
{
2012-03-31 02:40:54 +04:00
VALUE ary = ( VALUE ) a ;
* parse.y, compile.c, gc.c, insns.def, intern.h, iseq.c, node.h,
object.c, string.c, variable.c, vm_macro.def: revert private
instance variable feature, which is postponed until next major
release.
* marshal.c: TYPE_SYMBOL2 removed; MARSHAL_MINOR reverted back to
8th version.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@11813 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-02-23 05:49:41 +03:00
if ( rb_is_instance_id ( key ) ) {
2006-09-04 09:46:47 +04:00
rb_ary_push ( ary , ID2SYM ( key ) ) ;
1998-01-16 15:19:22 +03:00
}
return ST_CONTINUE ;
}
2003-12-28 09:33:07 +03:00
/*
* call - seq :
2010-05-18 01:07:33 +04:00
* obj . instance_variables - > array
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
* Returns an array of instance variable names for the receiver . Note
* that simply defining an accessor does not create the corresponding
* instance variable .
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
* class Fred
* attr_accessor : a1
* def initialize
* @ iv = 3
* end
* end
2008-03-09 04:04:46 +03:00
* Fred . new . instance_variables # = > [ : @ iv ]
2003-12-28 09:33:07 +03:00
*/
1998-01-16 15:19:22 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_obj_instance_variables ( VALUE obj )
1998-01-16 15:19:22 +03:00
{
1999-12-14 09:50:43 +03:00
VALUE ary ;
1998-01-16 15:19:22 +03:00
1999-12-14 09:50:43 +03:00
ary = rb_ary_new ( ) ;
2007-09-28 10:21:46 +04:00
rb_ivar_foreach ( obj , ivar_i , ary ) ;
1999-10-04 08:51:08 +04:00
return ary ;
1998-01-16 15:19:22 +03:00
}
2015-10-28 09:24:12 +03:00
# define rb_is_constant_id rb_is_const_id
# define rb_is_constant_name rb_is_const_name
# define id_for_var(obj, name, part, type) \
2024-01-19 10:03:38 +03:00
id_for_var_message ( obj , name , type , " '%1$s' is not allowed as " # part " " # type " variable name " )
2015-10-28 09:24:12 +03:00
# define id_for_var_message(obj, name, type, message) \
check_id_type ( obj , & ( name ) , rb_is_ # # type # # _id , rb_is_ # # type # # _name , message , strlen ( message ) )
static ID
check_id_type ( VALUE obj , VALUE * pname ,
int ( * valid_id_p ) ( ID ) , int ( * valid_name_p ) ( VALUE ) ,
const char * message , size_t message_len )
{
ID id = rb_check_id ( pname ) ;
VALUE name = * pname ;
if ( id ? ! valid_id_p ( id ) : ! valid_name_p ( name ) ) {
rb_name_err_raise_str ( rb_fstring_new ( message , message_len ) ,
obj , name ) ;
}
return id ;
}
2003-12-28 09:33:07 +03:00
/*
* call - seq :
2010-05-18 01:07:33 +04:00
* obj . remove_instance_variable ( symbol ) - > obj
2018-07-14 20:04:56 +03:00
* obj . remove_instance_variable ( string ) - > obj
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
* Removes the named instance variable from < i > obj < / i > , returning that
2024-03-09 15:58:49 +03:00
* variable ' s value . The name can be passed as a symbol or as a string .
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
* class Dummy
* attr_reader : var
* def initialize
* @ var = 99
* end
* def remove
* remove_instance_variable ( : @ var )
* end
* end
* d = Dummy . new
* d . var # = > 99
* d . remove # = > 99
* d . var # = > nil
*/
1998-01-16 15:19:22 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_obj_remove_instance_variable ( VALUE obj , VALUE name )
1998-01-16 15:19:22 +03:00
{
2015-10-28 09:24:12 +03:00
const ID id = id_for_var ( obj , name , an , instance ) ;
1998-01-16 15:19:22 +03:00
2022-10-03 18:14:32 +03:00
// Frozen check comes here because it's expected that we raise a
// NameError (from the id_for_var check) before we raise a FrozenError
* array.c, gc.c, hash.c, object.c, string.c, struct.c,
transcode.c, variable.c, vm.c, vm_insnhelper.c, vm_method.c:
replace calls to rb_error_frozen() with rb_check_frozen(). a
patch from Run Paint Run Run at [ruby-core:32014]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@29583 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2010-10-24 12:14:05 +04:00
rb_check_frozen ( obj ) ;
2022-10-03 18:14:32 +03:00
2023-11-01 20:13:51 +03:00
if ( id ) {
VALUE val = rb_ivar_delete ( obj , id , Qundef ) ;
2012-04-14 04:36:26 +04:00
2024-01-30 08:48:59 +03:00
if ( ! UNDEF_P ( val ) ) return val ;
2022-12-06 03:48:47 +03:00
}
2015-10-28 09:24:12 +03:00
rb_name_err_raise ( " instance variable %1$s not defined " ,
obj , name ) ;
2018-07-24 08:38:07 +03:00
UNREACHABLE_RETURN ( Qnil ) ;
1998-01-16 15:19:22 +03:00
}
2014-11-21 19:11:55 +03:00
NORETURN ( static void uninitialized_constant ( VALUE , VALUE ) ) ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
static void
2014-11-21 19:11:55 +03:00
uninitialized_constant ( VALUE klass , VALUE name )
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
{
2010-06-12 04:20:27 +04:00
if ( klass & & rb_class_real ( klass ) ! = rb_cObject )
2015-10-28 09:24:12 +03:00
rb_name_err_raise ( " uninitialized constant %2$s::%1$s " ,
klass , name ) ;
else
rb_name_err_raise ( " uninitialized constant %1$s " ,
klass , name ) ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2014-11-21 19:11:55 +03:00
VALUE
rb_const_missing ( VALUE klass , VALUE name )
2003-07-22 12:42:47 +04:00
{
2018-01-22 16:09:37 +03:00
VALUE value = rb_funcallv ( klass , idConst_missing , 1 , & name ) ;
2014-11-21 19:11:55 +03:00
rb_vm_inc_const_missing_count ( ) ;
return value ;
2003-07-22 12:42:47 +04:00
}
2003-12-28 09:33:07 +03:00
/*
* call - seq :
2010-05-18 01:07:33 +04:00
* mod . const_missing ( sym ) - > obj
2003-12-28 09:33:07 +03:00
*
2011-06-16 10:17:59 +04:00
* Invoked when a reference is made to an undefined constant in
* < i > mod < / i > . It is passed a symbol for the undefined constant , and
2024-03-09 15:58:49 +03:00
* returns a value to be used for that constant . For example , consider :
2009-02-15 05:45:31 +03:00
*
2011-06-16 10:17:59 +04:00
* def Foo . const_missing ( name )
* name # return the constant name as Symbol
* end
*
* Foo : : UNDEFINED_CONST # = > : UNDEFINED_CONST : symbol returned
*
2024-03-09 16:00:02 +03:00
* As the example above shows , + const_missing + is not required to create the
* missing constant in < i > mod < / i > , though that is often a side - effect . The
* caller gets its return value when triggered . If the constant is also defined ,
* further lookups won ' t hit + const_missing + and will return the value stored in
* the constant as usual . Otherwise , + const_missing + will be invoked again .
*
* In the next example , when a reference is made to an undefined constant ,
* + const_missing + attempts to load a file whose path is the lowercase version
* of the constant name ( thus class < code > Fred < / code > is assumed to be in file
* < code > fred . rb < / code > ) . If defined as a side - effect of loading the file , the
* method returns the value stored in the constant . This implements an autoload
* feature similar to Kernel # autoload and Module # autoload , though it differs in
* important ways .
2011-06-16 10:17:59 +04:00
*
* def Object . const_missing ( name )
* @ looked_for | | = { }
* str_name = name . to_s
2024-03-09 16:00:02 +03:00
* raise " Constant not found: #{name} " if @ looked_for [ str_name ]
2011-06-16 10:17:59 +04:00
* @ looked_for [ str_name ] = 1
* file = str_name . downcase
* require file
2024-03-09 16:00:02 +03:00
* const_get ( name , false )
2011-06-16 10:17:59 +04:00
* end
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
*/
2003-07-22 12:42:47 +04:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_mod_const_missing ( VALUE klass , VALUE name )
2003-07-22 12:42:47 +04:00
{
2023-04-19 05:40:16 +03:00
rb_execution_context_t * ec = GET_EC ( ) ;
VALUE ref = ec - > private_const_reference ;
2014-01-09 14:12:59 +04:00
rb_vm_pop_cfunc_frame ( ) ;
2018-07-06 16:56:58 +03:00
if ( ref ) {
2023-04-19 05:40:16 +03:00
ec - > private_const_reference = 0 ;
rb_name_err_raise ( " private constant %2$s::%1$s referenced " , ref , name ) ;
2018-07-06 16:56:58 +03:00
}
2014-11-21 19:11:55 +03:00
uninitialized_constant ( klass , name ) ;
2012-04-14 04:36:26 +04:00
2018-07-24 08:38:07 +03:00
UNREACHABLE_RETURN ( Qnil ) ;
2003-07-22 12:42:47 +04:00
}
2009-09-09 08:09:09 +04:00
static void
2022-05-16 15:50:02 +03:00
autoload_table_mark ( void * ptr )
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
{
2019-04-20 04:19:47 +03:00
rb_mark_tbl_no_pin ( ( st_table * ) ptr ) ;
2009-09-09 08:09:09 +04:00
}
static void
2022-05-16 15:50:02 +03:00
autoload_table_free ( void * ptr )
2009-09-09 08:09:09 +04:00
{
st_free_table ( ( st_table * ) ptr ) ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2009-09-09 08:09:09 +04:00
static size_t
2022-05-16 15:50:02 +03:00
autoload_table_memsize ( const void * ptr )
2009-09-09 08:09:09 +04:00
{
const st_table * tbl = ptr ;
2014-09-13 09:14:51 +04:00
return st_memsize ( tbl ) ;
2009-09-09 08:09:09 +04:00
}
2019-04-20 04:19:47 +03:00
static void
2022-05-16 15:50:02 +03:00
autoload_table_compact ( void * ptr )
2019-04-20 04:19:47 +03:00
{
rb_gc_update_tbl_refs ( ( st_table * ) ptr ) ;
}
2022-05-16 15:50:02 +03:00
static const rb_data_type_t autoload_table_type = {
" autoload_table " ,
{ autoload_table_mark , autoload_table_free , autoload_table_memsize , autoload_table_compact , } ,
2023-02-28 12:04:53 +03:00
0 , 0 , RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED
2009-09-09 08:09:09 +04:00
} ;
# define check_autoload_table(av) \
2022-05-16 15:50:02 +03:00
( struct st_table * ) rb_check_typeddata ( ( av ) , & autoload_table_type )
2009-09-09 08:09:09 +04:00
2011-08-31 12:28:19 +04:00
static VALUE
autoload_data ( VALUE mod , ID id )
{
struct st_table * tbl ;
st_data_t val ;
2022-11-01 00:05:37 +03:00
// If we are called with a non-origin ICLASS, fetch the autoload data from
// the original module.
if ( RB_TYPE_P ( mod , T_ICLASS ) ) {
if ( FL_TEST_RAW ( mod , RICLASS_IS_ORIGIN ) ) {
return 0 ;
2022-11-10 03:11:20 +03:00
}
else {
2022-11-01 00:05:37 +03:00
mod = RBASIC ( mod ) - > klass ;
}
}
RUBY_ASSERT ( RB_TYPE_P ( mod , T_CLASS ) | | RB_TYPE_P ( mod , T_MODULE ) ) ;
2022-05-16 15:50:02 +03:00
// Look up the instance variable table for `autoload`, then index into that table with the given constant name `id`.
2022-11-01 00:05:37 +03:00
2023-02-28 12:04:53 +03:00
VALUE tbl_value = rb_ivar_lookup ( mod , autoload , Qfalse ) ;
if ( ! RTEST ( tbl_value ) | | ! ( tbl = check_autoload_table ( tbl_value ) ) | | ! st_lookup ( tbl , ( st_data_t ) id , & val ) ) {
2022-05-16 15:50:02 +03:00
return 0 ;
2011-08-31 12:28:19 +04:00
}
2022-05-16 15:50:02 +03:00
2011-08-31 12:28:19 +04:00
return ( VALUE ) val ;
}
2022-05-16 15:50:02 +03:00
// Every autoload constant has exactly one instance of autoload_const, stored in `autoload_features`. Since multiple autoload constants can refer to the same file, every `autoload_const` refers to a de-duplicated `autoload_data`.
2018-05-10 11:54:26 +03:00
struct autoload_const {
2022-05-16 15:50:02 +03:00
// The linked list node of all constants which are loaded by the related autoload feature.
struct ccan_list_node cnode ; /* <=> autoload_data.constants */
// The shared "autoload_data" if multiple constants are defined from the same feature.
VALUE autoload_data_value ;
// The module we are loading a constant into.
VALUE module ;
// The name of the constant we are loading.
2022-05-21 15:32:41 +03:00
ID name ;
2022-05-16 15:50:02 +03:00
// The value of the constant (after it's loaded).
2018-05-10 11:54:26 +03:00
VALUE value ;
2022-05-16 15:50:02 +03:00
// The constant entry flags which need to be re-applied after autoloading the feature.
2018-05-10 11:54:26 +03:00
rb_const_flag_t flag ;
2022-05-16 15:50:02 +03:00
// The source file and line number that defined this constant (different from feature path).
VALUE file ;
2020-04-07 15:35:28 +03:00
int line ;
2018-05-10 11:54:26 +03:00
} ;
2022-05-16 15:50:02 +03:00
// Each `autoload_data` uniquely represents a specific feature which can be loaded, and a list of constants which it is able to define. We use a mutex to coordinate multiple threads trying to load the same feature.
struct autoload_data {
// The feature path to require to load this constant.
VALUE feature ;
// The mutex which is protecting autoloading this feature.
2022-05-08 01:22:58 +03:00
VALUE mutex ;
2015-10-29 04:14:45 +03:00
2022-05-16 15:50:02 +03:00
// The process fork serial number since the autoload mutex will become invalid on fork.
2018-04-20 06:22:26 +03:00
rb_serial_t fork_gen ;
2022-05-16 15:50:02 +03:00
// The linked list of all constants that are going to be loaded by this autoload.
2022-03-30 10:36:31 +03:00
struct ccan_list_head constants ; /* <=> autoload_const.cnode */
2011-08-31 12:28:19 +04:00
} ;
2019-04-20 04:19:47 +03:00
static void
2022-05-16 15:50:02 +03:00
autoload_data_compact ( void * ptr )
2019-04-20 04:19:47 +03:00
{
2022-05-16 15:50:02 +03:00
struct autoload_data * p = ptr ;
2019-05-18 12:23:47 +03:00
p - > feature = rb_gc_location ( p - > feature ) ;
2022-05-16 15:50:02 +03:00
p - > mutex = rb_gc_location ( p - > mutex ) ;
2019-04-20 04:19:47 +03:00
}
2011-08-31 12:28:19 +04:00
static void
2022-05-16 15:50:02 +03:00
autoload_data_mark ( void * ptr )
2011-08-31 12:28:19 +04:00
{
2022-05-16 15:50:02 +03:00
struct autoload_data * p = ptr ;
2018-05-10 11:54:26 +03:00
2019-08-12 23:09:21 +03:00
rb_gc_mark_movable ( p - > feature ) ;
2022-05-16 15:50:02 +03:00
rb_gc_mark_movable ( p - > mutex ) ;
2022-05-17 10:12:36 +03:00
}
static void
autoload_data_free ( void * ptr )
{
struct autoload_data * p = ptr ;
2024-01-06 01:14:16 +03:00
struct autoload_const * autoload_const , * next ;
ccan_list_for_each_safe ( & p - > constants , autoload_const , next , cnode ) {
ccan_list_del_init ( & autoload_const - > cnode ) ;
2022-05-17 10:12:36 +03:00
}
2024-01-06 01:14:16 +03:00
ruby_xfree ( p ) ;
2011-08-31 12:28:19 +04:00
}
static size_t
2022-05-16 15:50:02 +03:00
autoload_data_memsize ( const void * ptr )
2011-08-31 12:28:19 +04:00
{
2022-05-16 15:50:02 +03:00
return sizeof ( struct autoload_data ) ;
2011-08-31 12:28:19 +04:00
}
2022-05-16 15:50:02 +03:00
static const rb_data_type_t autoload_data_type = {
" autoload_data " ,
2022-05-17 10:12:36 +03:00
{ autoload_data_mark , autoload_data_free , autoload_data_memsize , autoload_data_compact } ,
2023-02-27 17:46:44 +03:00
0 , 0 , RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED
2018-05-10 11:54:26 +03:00
} ;
2019-05-31 23:25:24 +03:00
static void
2022-05-17 10:12:36 +03:00
autoload_const_compact ( void * ptr )
2019-05-31 23:25:24 +03:00
{
struct autoload_const * ac = ptr ;
2022-05-16 15:50:02 +03:00
ac - > module = rb_gc_location ( ac - > module ) ;
ac - > autoload_data_value = rb_gc_location ( ac - > autoload_data_value ) ;
2019-05-31 23:25:24 +03:00
ac - > value = rb_gc_location ( ac - > value ) ;
2020-04-07 15:35:28 +03:00
ac - > file = rb_gc_location ( ac - > file ) ;
2019-05-31 23:25:24 +03:00
}
2018-05-10 11:54:26 +03:00
static void
2022-05-17 10:12:36 +03:00
autoload_const_mark ( void * ptr )
2018-05-10 11:54:26 +03:00
{
struct autoload_const * ac = ptr ;
2022-05-16 15:50:02 +03:00
rb_gc_mark_movable ( ac - > module ) ;
rb_gc_mark_movable ( ac - > autoload_data_value ) ;
2019-08-12 23:09:21 +03:00
rb_gc_mark_movable ( ac - > value ) ;
2020-04-07 15:35:28 +03:00
rb_gc_mark_movable ( ac - > file ) ;
2018-05-10 11:54:26 +03:00
}
static size_t
2022-05-17 10:12:36 +03:00
autoload_const_memsize ( const void * ptr )
2018-05-10 11:54:26 +03:00
{
return sizeof ( struct autoload_const ) ;
}
2022-05-17 10:12:36 +03:00
static void
autoload_const_free ( void * ptr )
{
struct autoload_const * autoload_const = ptr ;
ccan_list_del ( & autoload_const - > cnode ) ;
ruby_xfree ( ptr ) ;
}
2018-05-10 11:54:26 +03:00
static const rb_data_type_t autoload_const_type = {
" autoload_const " ,
2022-05-17 10:12:36 +03:00
{ autoload_const_mark , autoload_const_free , autoload_const_memsize , autoload_const_compact , } ,
2014-12-01 09:38:04 +03:00
0 , 0 , RUBY_TYPED_FREE_IMMEDIATELY
2011-08-31 12:28:19 +04:00
} ;
2022-05-16 15:50:02 +03:00
static struct autoload_data *
2022-05-17 10:12:36 +03:00
get_autoload_data ( VALUE autoload_const_value , struct autoload_const * * autoload_const_pointer )
2018-04-20 06:22:26 +03:00
{
2022-05-17 10:12:36 +03:00
struct autoload_const * autoload_const = rb_check_typeddata ( autoload_const_value , & autoload_const_type ) ;
2023-02-27 17:46:44 +03:00
VALUE autoload_data_value = autoload_const - > autoload_data_value ;
struct autoload_data * autoload_data = rb_check_typeddata ( autoload_data_value , & autoload_data_type ) ;
2018-04-20 06:22:26 +03:00
/* do not reach across stack for ->state after forking: */
2022-05-17 10:12:36 +03:00
if ( autoload_data & & autoload_data - > fork_gen ! = GET_VM ( ) - > fork_gen ) {
2023-02-27 17:46:44 +03:00
RB_OBJ_WRITE ( autoload_data_value , & autoload_data - > mutex , Qnil ) ;
2022-05-17 10:12:36 +03:00
autoload_data - > fork_gen = 0 ;
2018-04-20 06:22:26 +03:00
}
2022-05-17 10:12:36 +03:00
if ( autoload_const_pointer ) * autoload_const_pointer = autoload_const ;
return autoload_data ;
2018-04-20 06:22:26 +03:00
}
2011-08-31 12:28:19 +04:00
2024-02-23 00:02:10 +03:00
void
2022-05-17 10:12:36 +03:00
rb_autoload ( VALUE module , ID name , const char * feature )
introduce rb_autoload_str to replace rb_autoload
rb_autoload_str may be safer by preventing premature GC. It
can also be more efficient by passing a pre-frozen string that
can be deduped using rb_fstring. Common autoload callers (e.g.
rubygems, rdoc) already use string literals as the file
argument.
There seems to be no reason to expose rb_autoload_str to the
public C API since autoload is not performance-critical.
Applications may declare autoloads in Ruby code or via
rb_funcall; so merely deprecate rb_autoload without exposing
rb_autoload_str to new users.
Running: valgrind -v ruby -rrdoc -rubygems -e exit
shows a minor memory reduction (32-bit userspace)
before:
in use at exit: 1,600,621 bytes in 28,819 blocks
total heap usage: 55,786 allocs, 26,967 frees, 6,693,790 bytes allocated
after:
in use at exit: 1,599,778 bytes in 28,789 blocks
total heap usage: 55,739 allocs, 26,950 frees, 6,692,973 bytes allocated
* include/ruby/intern.h (rb_autoload): deprecate
* internal.h (rb_autoload_str): declare
* load.c (rb_mod_autoload): use rb_autoload_str
* variable.c (rb_autoload): become compatibility wrapper
(rb_autoload_str): hoisted out from old rb_autoload
[ruby-core:71369] [Feature #11664]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@52909 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-06 15:28:09 +03:00
{
2022-05-17 10:12:36 +03:00
if ( ! feature | | ! * feature ) {
rb_raise ( rb_eArgError , " empty feature name " ) ;
introduce rb_autoload_str to replace rb_autoload
rb_autoload_str may be safer by preventing premature GC. It
can also be more efficient by passing a pre-frozen string that
can be deduped using rb_fstring. Common autoload callers (e.g.
rubygems, rdoc) already use string literals as the file
argument.
There seems to be no reason to expose rb_autoload_str to the
public C API since autoload is not performance-critical.
Applications may declare autoloads in Ruby code or via
rb_funcall; so merely deprecate rb_autoload without exposing
rb_autoload_str to new users.
Running: valgrind -v ruby -rrdoc -rubygems -e exit
shows a minor memory reduction (32-bit userspace)
before:
in use at exit: 1,600,621 bytes in 28,819 blocks
total heap usage: 55,786 allocs, 26,967 frees, 6,693,790 bytes allocated
after:
in use at exit: 1,599,778 bytes in 28,789 blocks
total heap usage: 55,739 allocs, 26,950 frees, 6,692,973 bytes allocated
* include/ruby/intern.h (rb_autoload): deprecate
* internal.h (rb_autoload_str): declare
* load.c (rb_mod_autoload): use rb_autoload_str
* variable.c (rb_autoload): become compatibility wrapper
(rb_autoload_str): hoisted out from old rb_autoload
[ruby-core:71369] [Feature #11664]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@52909 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-06 15:28:09 +03:00
}
2022-05-16 15:50:02 +03:00
2022-05-17 10:12:36 +03:00
rb_autoload_str ( module , name , rb_fstring_cstr ( feature ) ) ;
introduce rb_autoload_str to replace rb_autoload
rb_autoload_str may be safer by preventing premature GC. It
can also be more efficient by passing a pre-frozen string that
can be deduped using rb_fstring. Common autoload callers (e.g.
rubygems, rdoc) already use string literals as the file
argument.
There seems to be no reason to expose rb_autoload_str to the
public C API since autoload is not performance-critical.
Applications may declare autoloads in Ruby code or via
rb_funcall; so merely deprecate rb_autoload without exposing
rb_autoload_str to new users.
Running: valgrind -v ruby -rrdoc -rubygems -e exit
shows a minor memory reduction (32-bit userspace)
before:
in use at exit: 1,600,621 bytes in 28,819 blocks
total heap usage: 55,786 allocs, 26,967 frees, 6,693,790 bytes allocated
after:
in use at exit: 1,599,778 bytes in 28,789 blocks
total heap usage: 55,739 allocs, 26,950 frees, 6,692,973 bytes allocated
* include/ruby/intern.h (rb_autoload): deprecate
* internal.h (rb_autoload_str): declare
* load.c (rb_mod_autoload): use rb_autoload_str
* variable.c (rb_autoload): become compatibility wrapper
(rb_autoload_str): hoisted out from old rb_autoload
[ruby-core:71369] [Feature #11664]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@52909 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-06 15:28:09 +03:00
}
2022-04-08 10:44:03 +03:00
static void const_set ( VALUE klass , ID id , VALUE val ) ;
static void const_added ( VALUE klass , ID const_name ) ;
2022-05-16 15:50:02 +03:00
struct autoload_arguments {
VALUE module ;
ID name ;
2022-05-17 10:12:36 +03:00
VALUE feature ;
2022-05-16 15:50:02 +03:00
} ;
static VALUE
2022-05-17 10:12:36 +03:00
autoload_feature_lookup_or_create ( VALUE feature , struct autoload_data * * autoload_data_pointer )
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
{
2022-05-17 10:12:36 +03:00
RUBY_ASSERT_MUTEX_OWNED ( autoload_mutex ) ;
RUBY_ASSERT_CRITICAL_SECTION_ENTER ( ) ;
VALUE autoload_data_value = rb_hash_aref ( autoload_features , feature ) ;
struct autoload_data * autoload_data ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
2022-05-17 10:12:36 +03:00
if ( NIL_P ( autoload_data_value ) ) {
autoload_data_value = TypedData_Make_Struct ( 0 , struct autoload_data , & autoload_data_type , autoload_data ) ;
2023-02-27 17:46:44 +03:00
RB_OBJ_WRITE ( autoload_data_value , & autoload_data - > feature , feature ) ;
RB_OBJ_WRITE ( autoload_data_value , & autoload_data - > mutex , Qnil ) ;
2022-05-17 10:12:36 +03:00
ccan_list_head_init ( & autoload_data - > constants ) ;
if ( autoload_data_pointer ) * autoload_data_pointer = autoload_data ;
rb_hash_aset ( autoload_features , feature , autoload_data_value ) ;
2022-05-19 10:45:37 +03:00
}
else if ( autoload_data_pointer ) {
2022-05-17 10:12:36 +03:00
* autoload_data_pointer = rb_check_typeddata ( autoload_data_value , & autoload_data_type ) ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
introduce rb_autoload_str to replace rb_autoload
rb_autoload_str may be safer by preventing premature GC. It
can also be more efficient by passing a pre-frozen string that
can be deduped using rb_fstring. Common autoload callers (e.g.
rubygems, rdoc) already use string literals as the file
argument.
There seems to be no reason to expose rb_autoload_str to the
public C API since autoload is not performance-critical.
Applications may declare autoloads in Ruby code or via
rb_funcall; so merely deprecate rb_autoload without exposing
rb_autoload_str to new users.
Running: valgrind -v ruby -rrdoc -rubygems -e exit
shows a minor memory reduction (32-bit userspace)
before:
in use at exit: 1,600,621 bytes in 28,819 blocks
total heap usage: 55,786 allocs, 26,967 frees, 6,693,790 bytes allocated
after:
in use at exit: 1,599,778 bytes in 28,789 blocks
total heap usage: 55,739 allocs, 26,950 frees, 6,692,973 bytes allocated
* include/ruby/intern.h (rb_autoload): deprecate
* internal.h (rb_autoload_str): declare
* load.c (rb_mod_autoload): use rb_autoload_str
* variable.c (rb_autoload): become compatibility wrapper
(rb_autoload_str): hoisted out from old rb_autoload
[ruby-core:71369] [Feature #11664]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@52909 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-06 15:28:09 +03:00
2022-05-17 10:12:36 +03:00
RUBY_ASSERT_CRITICAL_SECTION_LEAVE ( ) ;
return autoload_data_value ;
}
2023-02-28 12:04:53 +03:00
static VALUE
2022-05-19 10:45:37 +03:00
autoload_table_lookup_or_create ( VALUE module )
{
2023-02-28 12:04:53 +03:00
VALUE autoload_table_value = rb_ivar_lookup ( module , autoload , Qfalse ) ;
if ( RTEST ( autoload_table_value ) ) {
return autoload_table_value ;
2022-11-10 03:11:20 +03:00
}
else {
2023-02-28 12:04:53 +03:00
autoload_table_value = TypedData_Wrap_Struct ( 0 , & autoload_table_type , NULL ) ;
2022-11-01 00:05:37 +03:00
rb_class_ivar_set ( module , autoload , autoload_table_value ) ;
2023-02-28 12:04:53 +03:00
RTYPEDDATA_DATA ( autoload_table_value ) = st_init_numtable ( ) ;
return autoload_table_value ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2022-05-16 15:50:02 +03:00
}
static VALUE
autoload_synchronized ( VALUE _arguments )
{
struct autoload_arguments * arguments = ( struct autoload_arguments * ) _arguments ;
rb_const_entry_t * constant_entry = rb_const_lookup ( arguments - > module , arguments - > name ) ;
2022-11-15 07:24:08 +03:00
if ( constant_entry & & ! UNDEF_P ( constant_entry - > value ) ) {
2022-05-16 15:50:02 +03:00
return Qfalse ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2011-08-31 12:35:27 +04:00
2022-05-16 15:50:02 +03:00
// Reset any state associated with any previous constant:
const_set ( arguments - > module , arguments - > name , Qundef ) ;
2023-02-28 12:04:53 +03:00
VALUE autoload_table_value = autoload_table_lookup_or_create ( arguments - > module ) ;
struct st_table * autoload_table = check_autoload_table ( autoload_table_value ) ;
2022-05-16 15:50:02 +03:00
// Ensure the string is uniqued since we use an identity lookup:
2022-05-17 10:12:36 +03:00
VALUE feature = rb_fstring ( arguments - > feature ) ;
2022-05-16 15:50:02 +03:00
2022-05-17 10:12:36 +03:00
struct autoload_data * autoload_data ;
VALUE autoload_data_value = autoload_feature_lookup_or_create ( feature , & autoload_data ) ;
2022-05-16 15:50:02 +03:00
{
struct autoload_const * autoload_const ;
VALUE autoload_const_value = TypedData_Make_Struct ( 0 , struct autoload_const , & autoload_const_type , autoload_const ) ;
autoload_const - > module = arguments - > module ;
autoload_const - > name = arguments - > name ;
autoload_const - > value = Qundef ;
autoload_const - > flag = CONST_PUBLIC ;
autoload_const - > autoload_data_value = autoload_data_value ;
ccan_list_add_tail ( & autoload_data - > constants , & autoload_const - > cnode ) ;
st_insert ( autoload_table , ( st_data_t ) arguments - > name , ( st_data_t ) autoload_const_value ) ;
2023-02-28 12:04:53 +03:00
RB_OBJ_WRITTEN ( autoload_table_value , Qundef , autoload_const_value ) ;
2018-05-10 11:54:26 +03:00
}
2022-05-16 15:50:02 +03:00
return Qtrue ;
}
void
2022-05-17 10:12:36 +03:00
rb_autoload_str ( VALUE module , ID name , VALUE feature )
2022-05-16 15:50:02 +03:00
{
2022-05-17 10:12:36 +03:00
if ( ! rb_is_const_id ( name ) ) {
rb_raise ( rb_eNameError , " autoload must be constant name: % " PRIsVALUE " " , QUOTE_ID ( name ) ) ;
2018-05-10 11:54:26 +03:00
}
2022-05-16 15:50:02 +03:00
2022-05-17 10:12:36 +03:00
Check_Type ( feature , T_STRING ) ;
if ( ! RSTRING_LEN ( feature ) ) {
rb_raise ( rb_eArgError , " empty feature name " ) ;
2018-05-10 11:54:26 +03:00
}
2022-04-08 10:44:03 +03:00
2022-05-16 15:50:02 +03:00
struct autoload_arguments arguments = {
2022-05-17 10:12:36 +03:00
. module = module ,
. name = name ,
. feature = feature ,
2022-05-16 15:50:02 +03:00
} ;
VALUE result = rb_mutex_synchronize ( autoload_mutex , autoload_synchronized , ( VALUE ) & arguments ) ;
if ( result = = Qtrue ) {
2022-05-17 10:12:36 +03:00
const_added ( module , name ) ;
2022-05-16 15:50:02 +03:00
}
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2011-08-31 12:28:19 +04:00
static void
2022-05-17 10:12:36 +03:00
autoload_delete ( VALUE module , ID name )
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
{
2022-05-17 10:12:36 +03:00
RUBY_ASSERT_CRITICAL_SECTION_ENTER ( ) ;
2022-11-01 00:05:37 +03:00
st_data_t load = 0 , key = name ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
2022-11-01 00:05:37 +03:00
RUBY_ASSERT ( RB_TYPE_P ( module , T_CLASS ) | | RB_TYPE_P ( module , T_MODULE ) ) ;
2023-02-28 12:04:53 +03:00
VALUE table_value = rb_ivar_lookup ( module , autoload , Qfalse ) ;
if ( RTEST ( table_value ) ) {
2022-11-01 00:05:37 +03:00
struct st_table * table = check_autoload_table ( table_value ) ;
2022-05-16 15:50:02 +03:00
2022-05-17 10:12:36 +03:00
st_delete ( table , & key , & load ) ;
2023-02-28 12:04:53 +03:00
RB_OBJ_WRITTEN ( table_value , load , Qundef ) ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
2021-10-09 00:54:26 +03:00
/* Qfalse can indicate already deleted */
if ( load ! = Qfalse ) {
2022-05-17 10:12:36 +03:00
struct autoload_const * autoload_const ;
2022-05-17 14:44:14 +03:00
struct autoload_data * autoload_data = get_autoload_data ( ( VALUE ) load , & autoload_const ) ;
VM_ASSERT ( autoload_data ) ;
VM_ASSERT ( ! ccan_list_empty ( & autoload_data - > constants ) ) ;
2022-05-17 10:12:36 +03:00
2021-10-09 00:54:26 +03:00
/*
* we must delete here to avoid " already initialized " warnings
* with parallel autoload . Using list_del_init here so list_del
2022-05-17 10:12:36 +03:00
* works in autoload_const_free
2021-10-09 00:54:26 +03:00
*/
2022-05-17 10:12:36 +03:00
ccan_list_del_init ( & autoload_const - > cnode ) ;
2018-05-10 04:19:20 +03:00
2022-05-17 14:44:14 +03:00
if ( ccan_list_empty ( & autoload_data - > constants ) ) {
rb_hash_delete ( autoload_features , autoload_data - > feature ) ;
}
2022-05-17 10:12:36 +03:00
// If the autoload table is empty, we can delete it.
if ( table - > num_entries = = 0 ) {
2022-11-01 00:05:37 +03:00
rb_attr_delete ( module , autoload ) ;
2021-10-09 00:54:26 +03:00
}
}
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2022-05-17 10:12:36 +03:00
RUBY_ASSERT_CRITICAL_SECTION_LEAVE ( ) ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2022-05-08 01:22:58 +03:00
static int
2022-05-17 06:47:01 +03:00
autoload_by_someone_else ( struct autoload_data * ele )
{
2022-05-16 15:50:02 +03:00
return ele - > mutex ! = Qnil & & ! rb_mutex_owned_p ( ele - > mutex ) ;
2022-05-08 01:22:58 +03:00
}
2011-08-31 12:28:19 +04:00
static VALUE
check_autoload_required ( VALUE mod , ID id , const char * * loadingpath )
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
{
2022-05-16 15:50:02 +03:00
VALUE autoload_const_value = autoload_data ( mod , id ) ;
struct autoload_data * autoload_data ;
2008-12-04 21:29:20 +03:00
const char * loading ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
2022-05-16 15:50:02 +03:00
if ( ! autoload_const_value | | ! ( autoload_data = get_autoload_data ( autoload_const_value , 0 ) ) ) {
2022-05-08 01:22:58 +03:00
return 0 ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2022-05-08 01:22:58 +03:00
2022-05-16 15:50:02 +03:00
VALUE feature = autoload_data - > feature ;
2017-05-13 00:52:04 +03:00
/*
* if somebody else is autoloading , we MUST wait for them , since
* rb_provide_feature can provide a feature before autoload_const_set
* completes . We must wait until autoload_const_set finishes in
* the other thread .
*/
2022-05-16 15:50:02 +03:00
if ( autoload_by_someone_else ( autoload_data ) ) {
return autoload_const_value ;
2017-05-13 00:52:04 +03:00
}
2022-05-16 15:50:02 +03:00
loading = RSTRING_PTR ( feature ) ;
2022-05-08 01:22:58 +03:00
2019-09-21 05:06:22 +03:00
if ( ! rb_feature_provided ( loading , & loading ) ) {
2022-05-16 15:50:02 +03:00
return autoload_const_value ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2022-05-08 01:22:58 +03:00
2009-05-16 08:49:26 +04:00
if ( loadingpath & & loading ) {
2022-05-08 01:22:58 +03:00
* loadingpath = loading ;
2022-05-16 15:50:02 +03:00
return autoload_const_value ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2022-05-08 01:22:58 +03:00
2008-12-04 21:29:20 +03:00
return 0 ;
}
2019-06-22 20:49:39 +03:00
static struct autoload_const * autoloading_const_entry ( VALUE mod , ID id ) ;
2018-12-13 12:12:07 +03:00
2023-03-07 08:34:31 +03:00
int
2018-02-13 12:36:35 +03:00
rb_autoloading_value ( VALUE mod , ID id , VALUE * value , rb_const_flag_t * flag )
2018-12-13 12:12:07 +03:00
{
struct autoload_const * ac = autoloading_const_entry ( mod , id ) ;
if ( ! ac ) return FALSE ;
if ( value ) {
* value = ac - > value ;
}
2022-05-15 07:07:12 +03:00
2018-12-13 12:12:07 +03:00
if ( flag ) {
* flag = ac - > flag ;
}
2022-05-15 07:07:12 +03:00
2018-12-13 12:12:07 +03:00
return TRUE ;
}
2022-05-08 01:22:58 +03:00
static int
2022-05-17 06:47:01 +03:00
autoload_by_current ( struct autoload_data * ele )
{
2022-05-16 15:50:02 +03:00
return ele - > mutex ! = Qnil & & rb_mutex_owned_p ( ele - > mutex ) ;
2022-05-08 01:22:58 +03:00
}
2022-05-15 07:07:12 +03:00
// If there is an autoloading constant and it has been set by the current
// execution context, return it. This allows threads which are loading code to
// refer to their own autoloaded constants.
2018-12-13 12:12:07 +03:00
struct autoload_const *
autoloading_const_entry ( VALUE mod , ID id )
2011-08-31 12:28:19 +04:00
{
2018-05-10 11:54:26 +03:00
VALUE load = autoload_data ( mod , id ) ;
2022-05-16 15:50:02 +03:00
struct autoload_data * ele ;
2018-05-10 11:54:26 +03:00
struct autoload_const * ac ;
2011-08-31 12:28:19 +04:00
2022-05-15 07:07:12 +03:00
// Find the autoloading state:
2018-05-10 11:54:26 +03:00
if ( ! load | | ! ( ele = get_autoload_data ( load , & ac ) ) ) {
2022-05-15 07:07:12 +03:00
// Couldn't be found:
2018-05-10 11:54:26 +03:00
return 0 ;
2011-08-31 12:28:19 +04:00
}
2018-05-10 11:54:26 +03:00
2022-05-15 07:07:12 +03:00
// Check if it's being loaded by the current thread/fiber:
2022-05-08 01:22:58 +03:00
if ( autoload_by_current ( ele ) ) {
2022-11-15 07:24:08 +03:00
if ( ! UNDEF_P ( ac - > value ) ) {
2018-12-13 12:12:07 +03:00
return ac ;
2022-05-08 01:22:58 +03:00
}
2011-08-31 12:28:19 +04:00
}
2022-05-08 01:22:58 +03:00
2011-08-31 12:28:19 +04:00
return 0 ;
}
2011-09-01 04:31:18 +04:00
static int
autoload_defined_p ( VALUE mod , ID id )
{
2014-08-04 05:12:53 +04:00
rb_const_entry_t * ce = rb_const_lookup ( mod , id ) ;
2011-09-01 04:31:18 +04:00
2022-05-15 07:07:12 +03:00
// If there is no constant or the constant is not undefined (special marker for autoloading):
2022-11-15 07:24:08 +03:00
if ( ! ce | | ! UNDEF_P ( ce - > value ) ) {
2022-05-15 07:07:12 +03:00
// We are not autoloading:
return 0 ;
2011-09-01 04:31:18 +04:00
}
2022-05-15 07:07:12 +03:00
// Otherwise check if there is an autoload in flight right now:
2018-02-13 12:36:35 +03:00
return ! rb_autoloading_value ( mod , id , NULL , NULL ) ;
2011-09-01 04:31:18 +04:00
}
2022-05-15 07:07:12 +03:00
static void const_tbl_update ( struct autoload_const * , int ) ;
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
2022-05-16 15:50:02 +03:00
struct autoload_load_arguments {
VALUE module ;
ID name ;
int flag ;
VALUE mutex ;
// The specific constant which triggered the autoload code to fire:
struct autoload_const * autoload_const ;
// The parent autoload data which is shared between multiple constants:
struct autoload_data * autoload_data ;
} ;
2012-02-14 07:10:11 +04:00
static VALUE
2020-04-07 14:20:34 +03:00
autoload_const_set ( struct autoload_const * ac )
2011-08-31 12:28:19 +04:00
{
2022-05-16 15:50:02 +03:00
check_before_mod_set ( ac - > module , ac - > name , ac - > value , " constant " ) ;
2020-12-19 21:23:41 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
2022-05-15 07:07:12 +03:00
const_tbl_update ( ac , true ) ;
2020-12-19 21:23:41 +03:00
}
RB_VM_LOCK_LEAVE ( ) ;
2022-05-08 01:22:58 +03:00
return 0 ; /* ignored */
2011-08-31 12:28:19 +04:00
}
static VALUE
2022-05-16 15:50:02 +03:00
autoload_load_needed ( VALUE _arguments )
2011-08-31 12:28:19 +04:00
{
2022-05-16 15:50:02 +03:00
struct autoload_load_arguments * arguments = ( struct autoload_load_arguments * ) _arguments ;
const char * loading = 0 , * src ;
if ( ! autoload_defined_p ( arguments - > module , arguments - > name ) ) {
return Qfalse ;
}
2022-05-21 15:32:41 +03:00
VALUE autoload_const_value = check_autoload_required ( arguments - > module , arguments - > name , & loading ) ;
if ( ! autoload_const_value ) {
2022-05-16 15:50:02 +03:00
return Qfalse ;
}
src = rb_sourcefile ( ) ;
if ( src & & loading & & strcmp ( src , loading ) = = 0 ) {
return Qfalse ;
}
struct autoload_const * autoload_const ;
2022-05-21 15:32:41 +03:00
struct autoload_data * autoload_data ;
if ( ! ( autoload_data = get_autoload_data ( autoload_const_value , & autoload_const ) ) ) {
2022-05-16 15:50:02 +03:00
return Qfalse ;
}
2015-10-29 02:59:45 +03:00
2022-06-29 15:59:39 +03:00
if ( NIL_P ( autoload_data - > mutex ) ) {
2023-02-27 17:46:44 +03:00
RB_OBJ_WRITE ( autoload_const - > autoload_data_value , & autoload_data - > mutex , rb_mutex_new ( ) ) ;
2022-05-21 15:32:41 +03:00
autoload_data - > fork_gen = GET_VM ( ) - > fork_gen ;
2022-05-16 15:50:02 +03:00
}
2022-05-21 15:32:41 +03:00
else if ( rb_mutex_owned_p ( autoload_data - > mutex ) ) {
2022-05-16 15:50:02 +03:00
return Qfalse ;
}
2022-05-21 15:32:41 +03:00
arguments - > mutex = autoload_data - > mutex ;
2022-05-16 15:50:02 +03:00
arguments - > autoload_const = autoload_const ;
2015-10-29 02:59:45 +03:00
2022-05-21 15:32:41 +03:00
return autoload_const_value ;
2011-08-31 12:28:19 +04:00
}
2022-05-16 15:50:02 +03:00
static VALUE
autoload_apply_constants ( VALUE _arguments )
{
2022-05-17 10:12:36 +03:00
RUBY_ASSERT_CRITICAL_SECTION_ENTER ( ) ;
2022-05-15 07:07:12 +03:00
2022-05-17 10:12:36 +03:00
struct autoload_load_arguments * arguments = ( struct autoload_load_arguments * ) _arguments ;
2022-05-15 07:07:12 +03:00
2022-06-23 10:02:33 +03:00
struct autoload_const * autoload_const = 0 ; // for ccan_container_off_var()
2022-05-26 03:44:01 +03:00
struct autoload_const * next ;
2022-05-16 15:50:02 +03:00
2022-05-26 03:44:01 +03:00
// We use safe iteration here because `autoload_const_set` will eventually invoke
// `autoload_delete` which will remove the constant from the linked list. In theory, once
// the `autoload_data->constants` linked list is empty, we can remove it.
2022-05-17 14:44:14 +03:00
2022-05-26 03:44:01 +03:00
// Iterate over all constants and assign them:
ccan_list_for_each_safe ( & arguments - > autoload_data - > constants , autoload_const , next , cnode ) {
2022-11-15 07:24:08 +03:00
if ( ! UNDEF_P ( autoload_const - > value ) ) {
2022-05-26 03:44:01 +03:00
autoload_const_set ( autoload_const ) ;
2022-05-16 15:50:02 +03:00
}
2015-04-13 10:54:39 +03:00
}
2015-10-29 02:59:45 +03:00
2022-05-17 10:12:36 +03:00
RUBY_ASSERT_CRITICAL_SECTION_LEAVE ( ) ;
2022-05-15 07:07:12 +03:00
2022-05-16 15:50:02 +03:00
return Qtrue ;
2017-05-07 07:26:37 +03:00
}
2022-05-16 15:50:02 +03:00
static VALUE
2022-05-26 03:44:01 +03:00
autoload_feature_require ( VALUE _arguments )
2022-05-16 15:50:02 +03:00
{
2022-05-26 03:44:01 +03:00
struct autoload_load_arguments * arguments = ( struct autoload_load_arguments * ) _arguments ;
struct autoload_const * autoload_const = arguments - > autoload_const ;
// We save this for later use in autoload_apply_constants:
arguments - > autoload_data = rb_check_typeddata ( autoload_const - > autoload_data_value , & autoload_data_type ) ;
VALUE result = rb_funcall ( rb_vm_top_self ( ) , rb_intern ( " require " ) , 1 , arguments - > autoload_data - > feature ) ;
if ( RTEST ( result ) ) {
return rb_mutex_synchronize ( autoload_mutex , autoload_apply_constants , _arguments ) ;
}
return result ;
2022-05-16 15:50:02 +03:00
}
2022-05-15 07:07:12 +03:00
static VALUE
2022-05-16 15:50:02 +03:00
autoload_try_load ( VALUE _arguments )
2008-12-04 21:29:20 +03:00
{
2022-05-15 07:07:12 +03:00
struct autoload_load_arguments * arguments = ( struct autoload_load_arguments * ) _arguments ;
2022-05-26 03:44:01 +03:00
VALUE result = autoload_feature_require ( _arguments ) ;
2019-08-12 06:53:37 +03:00
2022-05-16 15:50:02 +03:00
// After we loaded the feature, if the constant is not defined, we remove it completely:
rb_const_entry_t * ce = rb_const_lookup ( arguments - > module , arguments - > name ) ;
2022-05-08 01:22:58 +03:00
2022-11-15 07:24:08 +03:00
if ( ! ce | | UNDEF_P ( ce - > value ) ) {
2022-05-26 03:44:01 +03:00
result = Qfalse ;
2015-10-29 02:59:45 +03:00
2022-05-16 15:50:02 +03:00
rb_const_remove ( arguments - > module , arguments - > name ) ;
2022-06-18 11:29:52 +03:00
if ( arguments - > module = = rb_cObject ) {
rb_warning (
" Expected % " PRIsVALUE " to define % " PRIsVALUE " but it didn't " ,
arguments - > autoload_data - > feature ,
ID2SYM ( arguments - > name )
) ;
}
else {
rb_warning (
" Expected % " PRIsVALUE " to define % " PRIsVALUE " ::% " PRIsVALUE " but it didn't " ,
arguments - > autoload_data - > feature ,
arguments - > module ,
ID2SYM ( arguments - > name )
) ;
}
2022-05-19 10:45:37 +03:00
}
else {
2022-05-16 15:50:02 +03:00
// Otherwise, it was loaded, copy the flags from the autoload constant:
ce - > flag | = arguments - > flag ;
2015-11-10 00:18:42 +03:00
}
2017-05-07 07:26:37 +03:00
2022-05-26 03:44:01 +03:00
return result ;
2022-05-15 07:07:12 +03:00
}
VALUE
rb_autoload_load ( VALUE module , ID name )
{
rb_const_entry_t * ce = rb_const_lookup ( module , name ) ;
// We bail out as early as possible without any synchronisation:
2022-11-15 07:24:08 +03:00
if ( ! ce | | ! UNDEF_P ( ce - > value ) ) {
2022-05-15 07:07:12 +03:00
return Qfalse ;
}
2022-05-16 15:50:02 +03:00
// At this point, we assume there might be autoloading, so fail if it's ractor:
2022-05-15 07:07:12 +03:00
if ( UNLIKELY ( ! rb_ractor_main_p ( ) ) ) {
rb_raise ( rb_eRactorUnsafeError , " require by autoload on non-main Ractor is not supported (%s) " , rb_id2name ( name ) ) ;
}
2023-02-28 21:05:30 +03:00
// This state is stored on the stack and is used during the autoload process.
2022-05-26 03:44:01 +03:00
struct autoload_load_arguments arguments = { . module = module , . name = name , . mutex = Qnil } ;
2022-05-15 07:07:12 +03:00
// Figure out whether we can autoload the named constant:
2022-05-21 15:32:41 +03:00
VALUE autoload_const_value = rb_mutex_synchronize ( autoload_mutex , autoload_load_needed , ( VALUE ) & arguments ) ;
2022-05-15 07:07:12 +03:00
// This confirms whether autoloading is required or not:
2022-05-21 15:32:41 +03:00
if ( autoload_const_value = = Qfalse ) return autoload_const_value ;
2022-05-15 07:07:12 +03:00
2022-05-16 15:50:02 +03:00
arguments . flag = ce - > flag & ( CONST_DEPRECATED | CONST_VISIBILITY_MASK ) ;
2022-05-15 07:07:12 +03:00
2022-05-16 15:50:02 +03:00
// Only one thread will enter here at a time:
2022-05-25 14:12:54 +03:00
VALUE result = rb_mutex_synchronize ( arguments . mutex , autoload_try_load , ( VALUE ) & arguments ) ;
// If you don't guard this value, it's possible for the autoload constant to
// be freed by another thread which loads multiple constants, one of which
// resolves to the constant this thread is trying to load, so proteect this
// so that it is not freed until we are done with it in `autoload_try_load`:
RB_GC_GUARD ( autoload_const_value ) ;
return result ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_autoload_p ( VALUE mod , ID id )
2019-05-07 13:52:24 +03:00
{
2019-06-21 11:21:34 +03:00
return rb_autoload_at_p ( mod , id , TRUE ) ;
2019-05-07 13:52:24 +03:00
}
VALUE
2019-06-21 11:21:34 +03:00
rb_autoload_at_p ( VALUE mod , ID id , int recur )
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
{
2011-08-31 12:28:19 +04:00
VALUE load ;
2022-05-16 15:50:02 +03:00
struct autoload_data * ele ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
2011-08-31 12:28:19 +04:00
while ( ! autoload_defined_p ( mod , id ) ) {
2019-06-21 11:29:08 +03:00
if ( ! recur ) return Qnil ;
2011-04-14 16:23:32 +04:00
mod = RCLASS_SUPER ( mod ) ;
if ( ! mod ) return Qnil ;
}
2011-08-31 12:28:19 +04:00
load = check_autoload_required ( mod , id , 0 ) ;
2009-05-15 10:15:14 +04:00
if ( ! load ) return Qnil ;
2018-05-10 11:54:26 +03:00
return ( ele = get_autoload_data ( load , 0 ) ) ? ele - > feature : Qnil ;
* eval.c (ev_const_defined, ev_const_get), variable.c
(rb_const_get_at, rb_const_get, rb_mod_remove_const): use Qundef
as autoload marker. [ruby-dev:18103], [ruby-dev:18184]
* eval.c (rb_mod_autoload, rb_mod_autoload_p): new method;
Module#autoload, Module#autoload?.
* variable.c (rb_autoload, rb_autoload_load, rb_autoload_p):
manage autoload constants per classes/modules.
* variable.c (rb_const_defined_at, rb_const_defined): return false
for autoloading constants.
* class.c (rb_define_class, rb_define_module), eval.c (rb_eval),
variable.c (rb_mod_const_at, rb_const_assign): removed autoload
stuff.
* intern.h: prototypes; rb_autoload, rb_autoload_load,
rb_autoload_p.
* lib/optparse.rb (OptionParser::Switch::PlacedArgument::parse):
do not treat unmatched argument as an option.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@3874 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2003-05-29 04:11:11 +04:00
}
2023-03-07 08:34:31 +03:00
void
2016-05-15 04:57:28 +03:00
rb_const_warn_if_deprecated ( const rb_const_entry_t * ce , VALUE klass , ID id )
{
2019-12-17 08:52:15 +03:00
if ( RB_CONST_DEPRECATED_P ( ce ) & &
rb_warning_category_enabled_p ( RB_WARN_CATEGORY_DEPRECATED ) ) {
2016-05-15 04:57:28 +03:00
if ( klass = = rb_cObject ) {
2020-09-28 20:10:31 +03:00
rb_category_warn ( RB_WARN_CATEGORY_DEPRECATED , " constant ::% " PRIsVALUE " is deprecated " , QUOTE_ID ( id ) ) ;
2016-05-15 04:57:28 +03:00
}
else {
2020-09-28 20:10:31 +03:00
rb_category_warn ( RB_WARN_CATEGORY_DEPRECATED , " constant % " PRIsVALUE " ::% " PRIsVALUE " is deprecated " ,
2016-05-15 04:57:28 +03:00
rb_class_name ( klass ) , QUOTE_ID ( id ) ) ;
}
}
}
2003-06-20 11:11:44 +04:00
static VALUE
2011-01-28 20:57:27 +03:00
rb_const_get_0 ( VALUE klass , ID id , int exclude , int recurse , int visibility )
2016-06-19 04:54:42 +03:00
{
VALUE c = rb_const_search ( klass , id , exclude , recurse , visibility ) ;
2022-11-15 07:24:08 +03:00
if ( ! UNDEF_P ( c ) ) {
2020-03-09 20:22:11 +03:00
if ( UNLIKELY ( ! rb_ractor_main_p ( ) ) ) {
if ( ! rb_ractor_shareable_p ( c ) ) {
2020-12-21 12:06:28 +03:00
rb_raise ( rb_eRactorIsolationError , " can not access non-shareable objects in constant % " PRIsVALUE " ::%s by non-main Ractor. " , rb_class_path ( klass ) , rb_id2name ( id ) ) ;
2020-03-09 20:22:11 +03:00
}
}
return c ;
}
2016-06-19 04:54:42 +03:00
return rb_const_missing ( klass , ID2SYM ( id ) ) ;
}
static VALUE
2018-12-13 12:27:32 +03:00
rb_const_search_from ( VALUE klass , ID id , int exclude , int recurse , int visibility )
1998-01-16 15:13:05 +03:00
{
2021-06-18 00:47:11 +03:00
VALUE value , current ;
bool first_iteration = true ;
1998-01-16 15:13:05 +03:00
2021-06-18 00:47:11 +03:00
for ( current = klass ;
RTEST ( current ) ;
current = RCLASS_SUPER ( current ) , first_iteration = false ) {
VALUE tmp ;
2022-05-08 01:22:58 +03:00
VALUE am = 0 ;
rb_const_entry_t * ce ;
2014-08-04 05:12:53 +04:00
2021-06-18 00:47:11 +03:00
if ( ! first_iteration & & RCLASS_ORIGIN ( current ) ! = current ) {
// This item in the super chain has an origin iclass
// that comes later in the chain. Skip this item so
// prepended modules take precedence.
continue ;
}
// Do lookup in original class or module in case we are at an origin
// iclass in the chain.
tmp = current ;
if ( BUILTIN_TYPE ( tmp ) = = T_ICLASS ) tmp = RBASIC ( tmp ) - > klass ;
// Do the lookup. Loop in case of autoload.
2022-05-08 01:22:58 +03:00
while ( ( ce = rb_const_lookup ( tmp , id ) ) ) {
if ( visibility & & RB_CONST_PRIVATE_P ( ce ) ) {
GET_EC ( ) - > private_const_reference = tmp ;
return Qundef ;
}
rb_const_warn_if_deprecated ( ce , tmp , id ) ;
value = ce - > value ;
2022-11-15 07:24:08 +03:00
if ( UNDEF_P ( value ) ) {
2018-12-13 12:12:07 +03:00
struct autoload_const * ac ;
2022-05-08 01:22:58 +03:00
if ( am = = tmp ) break ;
am = tmp ;
2018-12-13 12:12:07 +03:00
ac = autoloading_const_entry ( tmp , id ) ;
if ( ac ) return ac - > value ;
2022-05-08 01:22:58 +03:00
rb_autoload_load ( tmp , id ) ;
continue ;
}
2019-06-22 19:47:40 +03:00
if ( exclude & & tmp = = rb_cObject ) {
2022-05-08 01:22:58 +03:00
goto not_found ;
}
return value ;
}
if ( ! recurse ) break ;
1998-01-16 15:13:05 +03:00
}
2006-12-31 18:02:22 +03:00
2018-07-06 16:56:58 +03:00
not_found :
GET_EC ( ) - > private_const_reference = 0 ;
2016-06-19 04:54:42 +03:00
return Qundef ;
1998-01-16 15:13:05 +03:00
}
2018-12-13 12:27:32 +03:00
static VALUE
rb_const_search ( VALUE klass , ID id , int exclude , int recurse , int visibility )
{
VALUE value ;
if ( klass = = rb_cObject ) exclude = FALSE ;
value = rb_const_search_from ( klass , id , exclude , recurse , visibility ) ;
2022-11-15 07:24:08 +03:00
if ( ! UNDEF_P ( value ) ) return value ;
2018-12-13 12:27:32 +03:00
if ( exclude ) return value ;
if ( BUILTIN_TYPE ( klass ) ! = T_MODULE ) return value ;
/* search global const too, if klass is a module */
return rb_const_search_from ( rb_cObject , id , FALSE , recurse , visibility ) ;
}
2003-06-20 11:11:44 +04:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_const_get_from ( VALUE klass , ID id )
2003-06-20 11:11:44 +04:00
{
2011-01-28 20:57:27 +03:00
return rb_const_get_0 ( klass , id , TRUE , TRUE , FALSE ) ;
2003-06-20 11:11:44 +04:00
}
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_const_get ( VALUE klass , ID id )
2003-06-20 11:11:44 +04:00
{
2011-01-28 20:57:27 +03:00
return rb_const_get_0 ( klass , id , FALSE , TRUE , FALSE ) ;
2003-08-28 12:35:43 +04:00
}
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_const_get_at ( VALUE klass , ID id )
2003-08-28 12:35:43 +04:00
{
2011-01-28 20:57:27 +03:00
return rb_const_get_0 ( klass , id , TRUE , FALSE , FALSE ) ;
}
2023-03-07 08:34:31 +03:00
VALUE
2011-01-28 20:57:27 +03:00
rb_public_const_get_from ( VALUE klass , ID id )
{
return rb_const_get_0 ( klass , id , TRUE , TRUE , TRUE ) ;
}
2023-03-07 08:34:31 +03:00
VALUE
2011-01-28 20:57:27 +03:00
rb_public_const_get_at ( VALUE klass , ID id )
{
return rb_const_get_0 ( klass , id , TRUE , FALSE , TRUE ) ;
2003-06-20 11:11:44 +04:00
}
2018-12-28 13:33:02 +03:00
NORETURN ( static void undefined_constant ( VALUE mod , VALUE name ) ) ;
static void
undefined_constant ( VALUE mod , VALUE name )
{
rb_name_err_raise ( " constant %2$s::%1$s not defined " ,
mod , name ) ;
}
2018-12-13 15:49:05 +03:00
static VALUE
rb_const_location_from ( VALUE klass , ID id , int exclude , int recurse , int visibility )
{
while ( RTEST ( klass ) ) {
2019-06-22 19:47:40 +03:00
rb_const_entry_t * ce ;
2018-12-13 15:49:05 +03:00
2019-06-22 19:47:40 +03:00
while ( ( ce = rb_const_lookup ( klass , id ) ) ) {
if ( visibility & & RB_CONST_PRIVATE_P ( ce ) ) {
return Qnil ;
}
if ( exclude & & klass = = rb_cObject ) {
goto not_found ;
}
2024-01-16 11:33:16 +03:00
if ( UNDEF_P ( ce - > value ) ) { // autoload
VALUE autoload_const_value = autoload_data ( klass , id ) ;
if ( RTEST ( autoload_const_value ) ) {
struct autoload_const * autoload_const ;
struct autoload_data * autoload_data = get_autoload_data ( autoload_const_value , & autoload_const ) ;
if ( ! UNDEF_P ( autoload_const - > value ) & & RTEST ( rb_mutex_owned_p ( autoload_data - > mutex ) ) ) {
return rb_assoc_new ( autoload_const - > file , INT2NUM ( autoload_const - > line ) ) ;
}
}
}
2019-06-22 19:47:40 +03:00
if ( NIL_P ( ce - > file ) ) return rb_ary_new ( ) ;
return rb_assoc_new ( ce - > file , INT2NUM ( ce - > line ) ) ;
}
if ( ! recurse ) break ;
klass = RCLASS_SUPER ( klass ) ;
2018-12-13 15:49:05 +03:00
}
not_found :
return Qnil ;
}
static VALUE
rb_const_location ( VALUE klass , ID id , int exclude , int recurse , int visibility )
{
VALUE loc ;
if ( klass = = rb_cObject ) exclude = FALSE ;
loc = rb_const_location_from ( klass , id , exclude , recurse , visibility ) ;
if ( ! NIL_P ( loc ) ) return loc ;
if ( exclude ) return loc ;
if ( BUILTIN_TYPE ( klass ) ! = T_MODULE ) return loc ;
/* search global const too, if klass is a module */
return rb_const_location_from ( rb_cObject , id , FALSE , recurse , visibility ) ;
}
VALUE
rb_const_source_location ( VALUE klass , ID id )
{
return rb_const_location ( klass , id , FALSE , TRUE , FALSE ) ;
}
2023-03-07 08:34:31 +03:00
VALUE
2018-12-13 15:49:05 +03:00
rb_const_source_location_at ( VALUE klass , ID id )
{
return rb_const_location ( klass , id , TRUE , FALSE , FALSE ) ;
}
2003-12-28 09:33:07 +03:00
/*
* call - seq :
2010-05-18 01:07:33 +04:00
* remove_const ( sym ) - > obj
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
* Removes the definition of the given constant , returning that
2012-03-14 08:48:02 +04:00
* constant ' s previous value . If that constant referred to
* a module , this will not change that module ' s name and can lead
* to confusion .
2003-12-28 09:33:07 +03:00
*/
1999-01-20 07:59:39 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_mod_remove_const ( VALUE mod , VALUE name )
1999-01-20 07:59:39 +03:00
{
2015-10-28 09:24:12 +03:00
const ID id = id_for_var ( mod , name , a , constant ) ;
1999-01-20 07:59:39 +03:00
2011-07-23 19:05:03 +04:00
if ( ! id ) {
2018-12-28 13:33:02 +03:00
undefined_constant ( mod , name ) ;
1999-01-20 07:59:39 +03:00
}
2010-03-12 01:15:11 +03:00
return rb_const_remove ( mod , id ) ;
}
VALUE
rb_const_remove ( VALUE mod , ID id )
{
VALUE val ;
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
rb_const_entry_t * ce ;
2010-03-12 01:15:11 +03:00
* array.c, gc.c, hash.c, object.c, string.c, struct.c,
transcode.c, variable.c, vm.c, vm_insnhelper.c, vm_method.c:
replace calls to rb_error_frozen() with rb_check_frozen(). a
patch from Run Paint Run Run at [ruby-core:32014]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@29583 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2010-10-24 12:14:05 +04:00
rb_check_frozen ( mod ) ;
2022-05-16 15:50:02 +03:00
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
ce = rb_const_lookup ( mod , id ) ;
if ( ! ce | | ! rb_id_table_delete ( RCLASS_CONST_TBL ( mod ) , id ) ) {
2022-05-16 15:50:02 +03:00
if ( rb_const_defined_at ( mod , id ) ) {
rb_name_err_raise ( " cannot remove %2$s::%1$s " , mod , ID2SYM ( id ) ) ;
}
2018-12-28 13:33:02 +03:00
undefined_constant ( mod , ID2SYM ( id ) ) ;
1999-01-20 07:59:39 +03:00
}
2009-09-13 17:18:35 +04:00
2022-03-31 18:04:25 +03:00
rb_clear_constant_cache_for_id ( id ) ;
2009-09-13 17:18:35 +04:00
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
val = ce - > value ;
2022-05-16 15:50:02 +03:00
2022-11-15 07:24:08 +03:00
if ( UNDEF_P ( val ) ) {
2022-05-16 15:50:02 +03:00
autoload_delete ( mod , id ) ;
val = Qnil ;
1999-01-20 07:59:39 +03:00
}
2022-05-16 15:50:02 +03:00
ruby_xfree ( ce ) ;
2009-09-13 17:18:35 +04:00
return val ;
1999-01-20 07:59:39 +03:00
}
2014-10-08 12:27:54 +04:00
static int
cv_i_update ( st_data_t * k , st_data_t * v , st_data_t a , int existing )
{
if ( existing ) return ST_STOP ;
* v = a ;
return ST_CONTINUE ;
}
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
static enum rb_id_table_iterator_result
sv_i ( ID key , VALUE v , void * a )
2001-06-05 11:19:39 +04:00
{
2012-03-31 02:40:54 +04:00
rb_const_entry_t * ce = ( rb_const_entry_t * ) v ;
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
st_table * tbl = a ;
2012-03-31 02:40:54 +04:00
2001-06-05 11:19:39 +04:00
if ( rb_is_const_id ( key ) ) {
2014-10-08 12:27:54 +04:00
st_update ( tbl , ( st_data_t ) key , cv_i_update , ( st_data_t ) ce ) ;
2001-06-05 11:19:39 +04:00
}
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
return ID_TABLE_CONTINUE ;
2001-06-05 11:19:39 +04:00
}
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
static enum rb_id_table_iterator_result
rb_local_constants_i ( ID const_name , VALUE const_value , void * ary )
2013-12-02 12:26:29 +04:00
{
2016-07-19 09:39:01 +03:00
if ( rb_is_const_id ( const_name ) & & ! RB_CONST_PRIVATE_P ( ( rb_const_entry_t * ) const_value ) ) {
2016-05-27 11:38:50 +03:00
rb_ary_push ( ( VALUE ) ary , ID2SYM ( const_name ) ) ;
}
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
return ID_TABLE_CONTINUE ;
2013-12-02 12:26:29 +04:00
}
static VALUE
rb_local_constants ( VALUE mod )
{
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
struct rb_id_table * tbl = RCLASS_CONST_TBL ( mod ) ;
2013-12-02 12:26:29 +04:00
VALUE ary ;
if ( ! tbl ) return rb_ary_new2 ( 0 ) ;
2020-12-19 21:23:41 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
ary = rb_ary_new2 ( rb_id_table_size ( tbl ) ) ;
rb_id_table_foreach ( tbl , rb_local_constants_i , ( void * ) ary ) ;
}
RB_VM_LOCK_LEAVE ( ) ;
2013-12-02 12:26:29 +04:00
return ary ;
}
2001-06-05 11:19:39 +04:00
void *
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_mod_const_at ( VALUE mod , void * data )
1998-01-16 15:19:22 +03:00
{
2001-06-05 11:19:39 +04:00
st_table * tbl = data ;
if ( ! tbl ) {
tbl = st_init_numtable ( ) ;
}
2010-10-26 21:27:21 +04:00
if ( RCLASS_CONST_TBL ( mod ) ) {
2020-12-19 21:23:41 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
rb_id_table_foreach ( RCLASS_CONST_TBL ( mod ) , sv_i , tbl ) ;
}
RB_VM_LOCK_LEAVE ( ) ;
1998-01-16 15:13:05 +03:00
}
2001-06-05 11:19:39 +04:00
return tbl ;
1998-01-16 15:19:22 +03:00
}
2001-06-05 11:19:39 +04:00
void *
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_mod_const_of ( VALUE mod , void * data )
1998-01-16 15:19:22 +03:00
{
2003-07-03 15:02:53 +04:00
VALUE tmp = mod ;
1998-01-16 15:19:22 +03:00
for ( ; ; ) {
2003-07-03 15:02:53 +04:00
data = rb_mod_const_at ( tmp , data ) ;
2007-09-28 10:21:46 +04:00
tmp = RCLASS_SUPER ( tmp ) ;
2003-07-03 15:02:53 +04:00
if ( ! tmp ) break ;
if ( tmp = = rb_cObject & & mod ! = rb_cObject ) break ;
1998-01-16 15:19:22 +03:00
}
2001-06-05 11:19:39 +04:00
return data ;
}
static int
2011-01-28 20:57:48 +03:00
list_i ( st_data_t key , st_data_t value , VALUE ary )
2001-06-05 11:19:39 +04:00
{
2011-01-28 20:57:48 +03:00
ID sym = ( ID ) key ;
rb_const_entry_t * ce = ( rb_const_entry_t * ) value ;
2014-10-08 12:27:51 +04:00
if ( RB_CONST_PUBLIC_P ( ce ) ) rb_ary_push ( ary , ID2SYM ( sym ) ) ;
2001-06-05 11:19:39 +04:00
return ST_CONTINUE ;
}
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_const_list ( void * data )
2001-06-05 11:19:39 +04:00
{
st_table * tbl = data ;
VALUE ary ;
if ( ! tbl ) return rb_ary_new2 ( 0 ) ;
ary = rb_ary_new2 ( tbl - > num_entries ) ;
2004-09-29 09:15:33 +04:00
st_foreach_safe ( tbl , list_i , ary ) ;
2002-03-20 14:15:19 +03:00
st_free_table ( tbl ) ;
2001-06-05 11:19:39 +04:00
1998-01-16 15:19:22 +03:00
return ary ;
1998-01-16 15:13:05 +03:00
}
2003-12-28 09:33:07 +03:00
/*
* call - seq :
2010-05-18 01:07:33 +04:00
* mod . constants ( inherit = true ) - > array
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
* Returns an array of the names of the constants accessible in
* < i > mod < / i > . This includes the names of constants in any included
2012-07-19 10:41:47 +04:00
* modules ( example at start of section ) , unless the < i > inherit < / i >
2006-12-04 18:19:33 +03:00
* parameter is set to < code > false < / code > .
*
2016-03-23 04:39:41 +03:00
* The implementation makes no guarantees about the order in which the
* constants are yielded .
*
2010-05-18 01:07:33 +04:00
* IO . constants . include ? ( : SYNC ) # = > true
* IO . constants ( false ) . include ? ( : SYNC ) # = > false
2006-12-04 18:19:33 +03:00
*
2019-03-28 06:33:36 +03:00
* Also see Module # const_defined ? .
2003-12-28 09:33:07 +03:00
*/
1999-11-17 10:30:37 +03:00
VALUE
2014-06-18 10:16:39 +04:00
rb_mod_constants ( int argc , const VALUE * argv , VALUE mod )
1999-11-17 10:30:37 +03:00
{
2019-11-25 09:05:53 +03:00
bool inherit = true ;
2006-12-04 18:19:33 +03:00
2018-12-06 10:49:24 +03:00
if ( rb_check_arity ( argc , 0 , 1 ) ) inherit = RTEST ( argv [ 0 ] ) ;
2013-12-02 12:26:29 +04:00
2018-12-06 10:49:24 +03:00
if ( inherit ) {
2013-12-02 12:26:29 +04:00
return rb_const_list ( rb_mod_const_of ( mod , 0 ) ) ;
2006-12-04 18:19:33 +03:00
}
else {
2013-12-02 12:26:29 +04:00
return rb_local_constants ( mod ) ;
2006-12-04 18:19:33 +03:00
}
1999-11-17 10:30:37 +03:00
}
2003-08-30 04:04:02 +04:00
static int
2011-01-28 20:57:27 +03:00
rb_const_defined_0 ( VALUE klass , ID id , int exclude , int recurse , int visibility )
2003-06-20 11:11:44 +04:00
{
2009-07-30 11:10:51 +04:00
VALUE tmp ;
2006-12-31 18:02:22 +03:00
int mod_retry = 0 ;
2014-08-04 05:12:53 +04:00
rb_const_entry_t * ce ;
2003-06-20 11:11:44 +04:00
2003-10-02 07:00:23 +04:00
tmp = klass ;
retry :
2003-06-20 11:11:44 +04:00
while ( tmp ) {
2014-08-04 05:12:53 +04:00
if ( ( ce = rb_const_lookup ( tmp , id ) ) ) {
2014-10-08 12:27:51 +04:00
if ( visibility & & RB_CONST_PRIVATE_P ( ce ) ) {
2011-01-28 20:57:27 +03:00
return ( int ) Qfalse ;
}
2022-11-15 07:24:08 +03:00
if ( UNDEF_P ( ce - > value ) & & ! check_autoload_required ( tmp , id , 0 ) & &
2018-02-13 12:36:35 +03:00
! rb_autoloading_value ( tmp , id , NULL , NULL ) )
2009-09-22 17:11:21 +04:00
return ( int ) Qfalse ;
2022-07-21 19:23:58 +03:00
2018-01-27 09:12:09 +03:00
if ( exclude & & tmp = = rb_cObject & & klass ! = rb_cObject ) {
return ( int ) Qfalse ;
}
2022-07-21 19:23:58 +03:00
2009-09-22 17:11:21 +04:00
return ( int ) Qtrue ;
2003-06-20 11:11:44 +04:00
}
2011-06-30 22:34:09 +04:00
if ( ! recurse ) break ;
2007-09-28 10:21:46 +04:00
tmp = RCLASS_SUPER ( tmp ) ;
2003-06-20 11:11:44 +04:00
}
2006-12-31 18:02:22 +03:00
if ( ! exclude & & ! mod_retry & & BUILTIN_TYPE ( klass ) = = T_MODULE ) {
mod_retry = 1 ;
tmp = rb_cObject ;
goto retry ;
2003-08-30 04:04:02 +04:00
}
2009-09-22 17:11:21 +04:00
return ( int ) Qfalse ;
2003-06-20 11:11:44 +04:00
}
1998-01-16 15:13:05 +03:00
int
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_const_defined_from ( VALUE klass , ID id )
1998-01-16 15:13:05 +03:00
{
2011-01-28 20:57:27 +03:00
return rb_const_defined_0 ( klass , id , TRUE , TRUE , FALSE ) ;
2003-08-30 04:04:02 +04:00
}
1999-01-20 07:59:39 +03:00
2003-08-30 04:04:02 +04:00
int
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_const_defined ( VALUE klass , ID id )
2003-08-30 04:04:02 +04:00
{
2011-01-28 20:57:27 +03:00
return rb_const_defined_0 ( klass , id , FALSE , TRUE , FALSE ) ;
2003-08-30 04:04:02 +04:00
}
int
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_const_defined_at ( VALUE klass , ID id )
2003-08-30 04:04:02 +04:00
{
2011-01-28 20:57:27 +03:00
return rb_const_defined_0 ( klass , id , TRUE , FALSE , FALSE ) ;
}
2023-03-07 08:34:31 +03:00
int
2011-01-28 20:57:27 +03:00
rb_public_const_defined_from ( VALUE klass , ID id )
{
return rb_const_defined_0 ( klass , id , TRUE , TRUE , TRUE ) ;
}
2011-08-03 07:47:10 +04:00
static void
2010-10-26 21:27:21 +04:00
check_before_mod_set ( VALUE klass , ID id , VALUE val , const char * dest )
1998-01-16 15:13:05 +03:00
{
* array.c, gc.c, hash.c, object.c, string.c, struct.c,
transcode.c, variable.c, vm.c, vm_insnhelper.c, vm_method.c:
replace calls to rb_error_frozen() with rb_check_frozen(). a
patch from Run Paint Run Run at [ruby-core:32014]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@29583 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2010-10-24 12:14:05 +04:00
rb_check_frozen ( klass ) ;
1998-01-16 15:13:05 +03:00
}
2000-12-08 10:10:38 +03:00
2019-04-19 00:37:01 +03:00
static void set_namespace_path ( VALUE named_namespace , VALUE name ) ;
2019-04-12 06:46:28 +03:00
static enum rb_id_table_iterator_result
2019-04-19 00:37:01 +03:00
set_namespace_path_i ( ID id , VALUE v , void * payload )
2019-04-12 06:46:28 +03:00
{
rb_const_entry_t * ce = ( rb_const_entry_t * ) v ;
VALUE value = ce - > value ;
VALUE parental_path = * ( ( VALUE * ) payload ) ;
2021-10-06 14:18:35 +03:00
if ( ! rb_is_const_id ( id ) | | ! rb_namespace_p ( value ) ) {
2019-04-12 06:46:28 +03:00
return ID_TABLE_CONTINUE ;
}
2023-01-11 17:23:03 +03:00
bool has_permanent_classpath ;
2019-04-12 06:46:28 +03:00
classname ( value , & has_permanent_classpath ) ;
if ( has_permanent_classpath ) {
return ID_TABLE_CONTINUE ;
}
2019-04-19 00:37:01 +03:00
set_namespace_path ( value , build_const_path ( parental_path , id ) ) ;
2023-01-11 17:23:03 +03:00
if ( ! RCLASS_EXT ( value ) - > permanent_classpath ) {
RCLASS_SET_CLASSPATH ( value , 0 , false ) ;
}
2019-04-12 06:46:28 +03:00
return ID_TABLE_CONTINUE ;
}
/*
* Assign permanent classpaths to all namespaces that are directly or indirectly
* nested under + named_namespace + . + named_namespace + must have a permanent
* classpath .
*/
static void
2019-04-19 00:37:01 +03:00
set_namespace_path ( VALUE named_namespace , VALUE namespace_path )
2019-04-12 06:46:28 +03:00
{
struct rb_id_table * const_table = RCLASS_CONST_TBL ( named_namespace ) ;
2020-12-19 21:23:41 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
2023-01-11 17:23:03 +03:00
RCLASS_SET_CLASSPATH ( named_namespace , namespace_path , true ) ;
2020-12-19 21:23:41 +03:00
if ( const_table ) {
rb_id_table_foreach ( const_table , set_namespace_path_i , & namespace_path ) ;
}
2019-04-12 06:46:28 +03:00
}
2020-12-19 21:23:41 +03:00
RB_VM_LOCK_LEAVE ( ) ;
2019-04-12 06:46:28 +03:00
}
2021-05-22 13:04:01 +03:00
static void
const_added ( VALUE klass , ID const_name )
{
if ( GET_VM ( ) - > running ) {
2022-01-14 18:22:57 +03:00
VALUE name = ID2SYM ( const_name ) ;
rb_funcallv ( klass , idConst_added , 1 , & name ) ;
2021-05-22 13:04:01 +03:00
}
}
2022-04-08 10:44:03 +03:00
static void
const_set ( VALUE klass , ID id , VALUE val )
2000-02-18 09:59:36 +03:00
{
2010-10-26 21:27:32 +04:00
rb_const_entry_t * ce ;
* sprintf.c (rb_str_format): allow %c to print one character
string (e.g. ?x).
* lib/tempfile.rb (Tempfile::make_tmpname): put dot between
basename and pid. [ruby-talk:196272]
* parse.y (do_block): remove -> style block.
* parse.y (parser_yylex): remove tLAMBDA_ARG.
* eval.c (rb_call0): binding for the return event hook should have
consistent scope. [ruby-core:07928]
* eval.c (proc_invoke): return behavior should depend whether it
is surrounded by a lambda or a mere block.
* eval.c (formal_assign): handles post splat arguments.
* eval.c (rb_call0): ditto.
* st.c (strhash): use FNV-1a hash.
* parse.y (parser_yylex): removed experimental ';;' terminator.
* eval.c (rb_node_arity): should be aware of post splat arguments.
* eval.c (rb_proc_arity): ditto.
* parse.y (f_args): syntax rule enhanced to support arguments
after the splat.
* parse.y (block_param): ditto for block parameters.
* parse.y (f_post_arg): mandatory formal arguments after the splat
argument.
* parse.y (new_args_gen): generate nodes for mandatory formal
arguments after the splat argument.
* eval.c (rb_eval): dispatch mandatory formal arguments after the
splat argument.
* parse.y (args): allow more than one splat in the argument list.
* parse.y (method_call): allow aref [] to accept all kind of
method argument, including assocs, splat, and block argument.
* eval.c (SETUP_ARGS0): prepare block argument as well.
* lib/mathn.rb (Integer): remove Integer#gcd2. [ruby-core:07931]
* eval.c (error_line): print receivers true/false/nil specially.
* eval.c (rb_proc_yield): handles parameters in yield semantics.
* eval.c (nil_yield): gives LocalJumpError to denote no block
error.
* io.c (rb_io_getc): now takes one-character string.
* string.c (rb_str_hash): use FNV-1a hash from Fowler/Noll/Vo
hashing algorithm.
* string.c (rb_str_aref): str[0] now returns 1 character string,
instead of a fixnum. [Ruby2]
* parse.y (parser_yylex): ?c now returns 1 character string,
instead of a fixnum. [Ruby2]
* string.c (rb_str_aset): no longer support fixnum insertion.
* eval.c (umethod_bind): should not update original class.
[ruby-dev:28636]
* eval.c (ev_const_get): should support constant access from
within instance_eval(). [ruby-dev:28327]
* time.c (time_timeval): should round for usec floating
number. [ruby-core:07896]
* time.c (time_add): ditto.
* dir.c (sys_warning): should not call a vararg function
rb_sys_warning() indirectly. [ruby-core:07886]
* numeric.c (flo_divmod): the first element of Float#divmod should
be an integer. [ruby-dev:28589]
* test/ruby/test_float.rb: add tests for divmod, div, modulo and remainder.
* re.c (rb_reg_initialize): should not allow modifying literal
regexps. frozen check moved from rb_reg_initialize_m as well.
* re.c (rb_reg_initialize): should not modify untainted objects in
safe levels higher than 3.
* re.c (rb_memcmp): type change from char* to const void*.
* dir.c (dir_close): should not close untainted dir stream.
* dir.c (GetDIR): add tainted/frozen check for each dir operation.
* lib/rdoc/parsers/parse_rb.rb (RDoc::RubyParser::parse_symbol_arg):
typo fixed. a patch from Florian Gross <florg at florg.net>.
* eval.c (EXEC_EVENT_HOOK): trace_func may remove itself from
event_hooks. no guarantee for arbitrary hook deletion.
[ruby-dev:28632]
* util.c (ruby_strtod): differ addition to minimize error.
[ruby-dev:28619]
* util.c (ruby_strtod): should not raise ERANGE when the input
string does not have any digits. [ruby-dev:28629]
* eval.c (proc_invoke): should restore old ruby_frame->block.
thanks to ts <decoux at moulon.inra.fr>. [ruby-core:07833]
also fix [ruby-dev:28614] as well.
* signal.c (trap): sig should be less then NSIG. Coverity found
this bug. a patch from Kevin Tew <tewk at tewk.com>.
[ruby-core:07823]
* math.c (math_log2): add new method inspired by
[ruby-talk:191237].
* math.c (math_log): add optional base argument to Math::log().
[ruby-talk:191308]
* ext/syck/emitter.c (syck_scan_scalar): avoid accessing
uninitialized array element. a patch from Pat Eyler
<rubypate at gmail.com>. [ruby-core:07809]
* array.c (rb_ary_fill): initialize local variables first. a
patch from Pat Eyler <rubypate at gmail.com>. [ruby-core:07810]
* ext/syck/yaml2byte.c (syck_yaml2byte_handler): need to free
type_tag. a patch from Pat Eyler <rubypate at gmail.com>.
[ruby-core:07808]
* ext/socket/socket.c (make_hostent_internal): accept ai_family
check from Sam Roberts <sroberts at uniserve.com>.
[ruby-core:07691]
* util.c (ruby_strtod): should not cut off 18 digits for no
reason. [ruby-core:07796]
* array.c (rb_ary_fill): internalize local variable "beg" to
pacify Coverity. [ruby-core:07770]
* pack.c (pack_unpack): now supports CRLF newlines. a patch from
<tommy at tmtm.org>. [ruby-dev:28601]
* applied code clean-up patch from Stefan Huehner
<stefan at huehner.org>. [ruby-core:07764]
* lib/jcode.rb (String::tr_s): should have translated non
squeezing character sequence (i.e. a character) as well. thanks
to Hiroshi Ichikawa <gimite at gimite.ddo.jp> [ruby-list:42090]
* ext/socket/socket.c: document update patch from Sam Roberts
<sroberts at uniserve.com>. [ruby-core:07701]
* lib/mathn.rb (Integer): need not to remove gcd2. a patch from
NARUSE, Yui <naruse at airemix.com>. [ruby-dev:28570]
* parse.y (arg): too much NEW_LIST()
* eval.c (SETUP_ARGS0): remove unnecessary access to nd_alen.
* eval.c (rb_eval): use ARGSCAT for NODE_OP_ASGN1.
[ruby-dev:28585]
* parse.y (arg): use NODE_ARGSCAT for placeholder.
* lib/getoptlong.rb (GetoptLong::get): RDoc update patch from
mathew <meta at pobox.com>. [ruby-core:07738]
* variable.c (rb_const_set): raise error when no target klass is
supplied. [ruby-dev:28582]
* prec.c (prec_prec_f): documentation patch from
<gerardo.santana at gmail.com>. [ruby-core:07689]
* bignum.c (rb_big_pow): second operand may be too big even if
it's a Fixnum. [ruby-talk:187984]
* README.EXT: update symbol description. [ruby-talk:188104]
* COPYING: explicitly note GPLv2. [ruby-talk:187922]
* parse.y: remove some obsolete syntax rules (unparenthesized
method calls in argument list).
* eval.c (rb_call0): insecure calling should be checked for non
NODE_SCOPE method invocations too.
* eval.c (rb_alias): should preserve the current safe level as
well as method definition.
* process.c (rb_f_sleep): remove RDoc description about SIGALRM
which is not valid on the current implementation. [ruby-dev:28464]
Thu Mar 23 21:40:47 2006 K.Kosako <sndgk393 AT ybb.ne.jp>
* eval.c (method_missing): should support argument splat in
super. a bug in combination of super, splat and
method_missing. [ruby-talk:185438]
* configure.in: Solaris SunPro compiler -rapth patch from
<kuwa at labs.fujitsu.com>. [ruby-dev:28443]
* configure.in: remove enable_rpath=no for Solaris.
[ruby-dev:28440]
* ext/win32ole/win32ole.c (ole_val2olevariantdata): change behavior
of converting OLE Variant object with VT_ARRAY|VT_UI1 and Ruby
String object.
* ruby.1: a clarification patch from David Lutterkort
<dlutter at redhat.com>. [ruby-core:7508]
* lib/rdoc/ri/ri_paths.rb (RI::Paths): adding paths from rubygems
directories. a patch from Eric Hodel <drbrain at segment7.net>.
[ruby-core:07423]
* eval.c (rb_clear_cache_by_class): clearing wrong cache.
* ext/extmk.rb: use :remove_destination to install extension libraries
to avoid SEGV. [ruby-dev:28417]
* eval.c (rb_thread_fd_writable): should not re-schedule output
from KILLED thread (must be error printing).
* array.c (rb_ary_flatten_bang): allow specifying recursion
level. [ruby-talk:182170]
* array.c (rb_ary_flatten): ditto.
* gc.c (add_heap): a heap_slots may overflow. a patch from Stefan
Weil <weil at mail.berlios.de>.
* eval.c (rb_call): use separate cache for fcall/vcall
invocation.
* eval.c (rb_eval): NODE_FCALL, NODE_VCALL can call local
functions.
* eval.c (rb_mod_local): a new method to specify newly added
visibility "local".
* eval.c (search_method): search for local methods which are
visible only from the current class.
* class.c (rb_class_local_methods): a method to list local methods.
* object.c (Init_Object): add BasicObject class as a top level
BlankSlate class.
* ruby.h (SYM2ID): should not cast to signed long.
[ruby-core:07414]
* class.c (rb_include_module): allow module duplication.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@10235 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2006-06-10 01:20:17 +04:00
if ( NIL_P ( klass ) ) {
2022-04-08 10:44:03 +03:00
rb_raise ( rb_eTypeError , " no class/module to define constant % " PRIsVALUE " " ,
QUOTE_ID ( id ) ) ;
* sprintf.c (rb_str_format): allow %c to print one character
string (e.g. ?x).
* lib/tempfile.rb (Tempfile::make_tmpname): put dot between
basename and pid. [ruby-talk:196272]
* parse.y (do_block): remove -> style block.
* parse.y (parser_yylex): remove tLAMBDA_ARG.
* eval.c (rb_call0): binding for the return event hook should have
consistent scope. [ruby-core:07928]
* eval.c (proc_invoke): return behavior should depend whether it
is surrounded by a lambda or a mere block.
* eval.c (formal_assign): handles post splat arguments.
* eval.c (rb_call0): ditto.
* st.c (strhash): use FNV-1a hash.
* parse.y (parser_yylex): removed experimental ';;' terminator.
* eval.c (rb_node_arity): should be aware of post splat arguments.
* eval.c (rb_proc_arity): ditto.
* parse.y (f_args): syntax rule enhanced to support arguments
after the splat.
* parse.y (block_param): ditto for block parameters.
* parse.y (f_post_arg): mandatory formal arguments after the splat
argument.
* parse.y (new_args_gen): generate nodes for mandatory formal
arguments after the splat argument.
* eval.c (rb_eval): dispatch mandatory formal arguments after the
splat argument.
* parse.y (args): allow more than one splat in the argument list.
* parse.y (method_call): allow aref [] to accept all kind of
method argument, including assocs, splat, and block argument.
* eval.c (SETUP_ARGS0): prepare block argument as well.
* lib/mathn.rb (Integer): remove Integer#gcd2. [ruby-core:07931]
* eval.c (error_line): print receivers true/false/nil specially.
* eval.c (rb_proc_yield): handles parameters in yield semantics.
* eval.c (nil_yield): gives LocalJumpError to denote no block
error.
* io.c (rb_io_getc): now takes one-character string.
* string.c (rb_str_hash): use FNV-1a hash from Fowler/Noll/Vo
hashing algorithm.
* string.c (rb_str_aref): str[0] now returns 1 character string,
instead of a fixnum. [Ruby2]
* parse.y (parser_yylex): ?c now returns 1 character string,
instead of a fixnum. [Ruby2]
* string.c (rb_str_aset): no longer support fixnum insertion.
* eval.c (umethod_bind): should not update original class.
[ruby-dev:28636]
* eval.c (ev_const_get): should support constant access from
within instance_eval(). [ruby-dev:28327]
* time.c (time_timeval): should round for usec floating
number. [ruby-core:07896]
* time.c (time_add): ditto.
* dir.c (sys_warning): should not call a vararg function
rb_sys_warning() indirectly. [ruby-core:07886]
* numeric.c (flo_divmod): the first element of Float#divmod should
be an integer. [ruby-dev:28589]
* test/ruby/test_float.rb: add tests for divmod, div, modulo and remainder.
* re.c (rb_reg_initialize): should not allow modifying literal
regexps. frozen check moved from rb_reg_initialize_m as well.
* re.c (rb_reg_initialize): should not modify untainted objects in
safe levels higher than 3.
* re.c (rb_memcmp): type change from char* to const void*.
* dir.c (dir_close): should not close untainted dir stream.
* dir.c (GetDIR): add tainted/frozen check for each dir operation.
* lib/rdoc/parsers/parse_rb.rb (RDoc::RubyParser::parse_symbol_arg):
typo fixed. a patch from Florian Gross <florg at florg.net>.
* eval.c (EXEC_EVENT_HOOK): trace_func may remove itself from
event_hooks. no guarantee for arbitrary hook deletion.
[ruby-dev:28632]
* util.c (ruby_strtod): differ addition to minimize error.
[ruby-dev:28619]
* util.c (ruby_strtod): should not raise ERANGE when the input
string does not have any digits. [ruby-dev:28629]
* eval.c (proc_invoke): should restore old ruby_frame->block.
thanks to ts <decoux at moulon.inra.fr>. [ruby-core:07833]
also fix [ruby-dev:28614] as well.
* signal.c (trap): sig should be less then NSIG. Coverity found
this bug. a patch from Kevin Tew <tewk at tewk.com>.
[ruby-core:07823]
* math.c (math_log2): add new method inspired by
[ruby-talk:191237].
* math.c (math_log): add optional base argument to Math::log().
[ruby-talk:191308]
* ext/syck/emitter.c (syck_scan_scalar): avoid accessing
uninitialized array element. a patch from Pat Eyler
<rubypate at gmail.com>. [ruby-core:07809]
* array.c (rb_ary_fill): initialize local variables first. a
patch from Pat Eyler <rubypate at gmail.com>. [ruby-core:07810]
* ext/syck/yaml2byte.c (syck_yaml2byte_handler): need to free
type_tag. a patch from Pat Eyler <rubypate at gmail.com>.
[ruby-core:07808]
* ext/socket/socket.c (make_hostent_internal): accept ai_family
check from Sam Roberts <sroberts at uniserve.com>.
[ruby-core:07691]
* util.c (ruby_strtod): should not cut off 18 digits for no
reason. [ruby-core:07796]
* array.c (rb_ary_fill): internalize local variable "beg" to
pacify Coverity. [ruby-core:07770]
* pack.c (pack_unpack): now supports CRLF newlines. a patch from
<tommy at tmtm.org>. [ruby-dev:28601]
* applied code clean-up patch from Stefan Huehner
<stefan at huehner.org>. [ruby-core:07764]
* lib/jcode.rb (String::tr_s): should have translated non
squeezing character sequence (i.e. a character) as well. thanks
to Hiroshi Ichikawa <gimite at gimite.ddo.jp> [ruby-list:42090]
* ext/socket/socket.c: document update patch from Sam Roberts
<sroberts at uniserve.com>. [ruby-core:07701]
* lib/mathn.rb (Integer): need not to remove gcd2. a patch from
NARUSE, Yui <naruse at airemix.com>. [ruby-dev:28570]
* parse.y (arg): too much NEW_LIST()
* eval.c (SETUP_ARGS0): remove unnecessary access to nd_alen.
* eval.c (rb_eval): use ARGSCAT for NODE_OP_ASGN1.
[ruby-dev:28585]
* parse.y (arg): use NODE_ARGSCAT for placeholder.
* lib/getoptlong.rb (GetoptLong::get): RDoc update patch from
mathew <meta at pobox.com>. [ruby-core:07738]
* variable.c (rb_const_set): raise error when no target klass is
supplied. [ruby-dev:28582]
* prec.c (prec_prec_f): documentation patch from
<gerardo.santana at gmail.com>. [ruby-core:07689]
* bignum.c (rb_big_pow): second operand may be too big even if
it's a Fixnum. [ruby-talk:187984]
* README.EXT: update symbol description. [ruby-talk:188104]
* COPYING: explicitly note GPLv2. [ruby-talk:187922]
* parse.y: remove some obsolete syntax rules (unparenthesized
method calls in argument list).
* eval.c (rb_call0): insecure calling should be checked for non
NODE_SCOPE method invocations too.
* eval.c (rb_alias): should preserve the current safe level as
well as method definition.
* process.c (rb_f_sleep): remove RDoc description about SIGALRM
which is not valid on the current implementation. [ruby-dev:28464]
Thu Mar 23 21:40:47 2006 K.Kosako <sndgk393 AT ybb.ne.jp>
* eval.c (method_missing): should support argument splat in
super. a bug in combination of super, splat and
method_missing. [ruby-talk:185438]
* configure.in: Solaris SunPro compiler -rapth patch from
<kuwa at labs.fujitsu.com>. [ruby-dev:28443]
* configure.in: remove enable_rpath=no for Solaris.
[ruby-dev:28440]
* ext/win32ole/win32ole.c (ole_val2olevariantdata): change behavior
of converting OLE Variant object with VT_ARRAY|VT_UI1 and Ruby
String object.
* ruby.1: a clarification patch from David Lutterkort
<dlutter at redhat.com>. [ruby-core:7508]
* lib/rdoc/ri/ri_paths.rb (RI::Paths): adding paths from rubygems
directories. a patch from Eric Hodel <drbrain at segment7.net>.
[ruby-core:07423]
* eval.c (rb_clear_cache_by_class): clearing wrong cache.
* ext/extmk.rb: use :remove_destination to install extension libraries
to avoid SEGV. [ruby-dev:28417]
* eval.c (rb_thread_fd_writable): should not re-schedule output
from KILLED thread (must be error printing).
* array.c (rb_ary_flatten_bang): allow specifying recursion
level. [ruby-talk:182170]
* array.c (rb_ary_flatten): ditto.
* gc.c (add_heap): a heap_slots may overflow. a patch from Stefan
Weil <weil at mail.berlios.de>.
* eval.c (rb_call): use separate cache for fcall/vcall
invocation.
* eval.c (rb_eval): NODE_FCALL, NODE_VCALL can call local
functions.
* eval.c (rb_mod_local): a new method to specify newly added
visibility "local".
* eval.c (search_method): search for local methods which are
visible only from the current class.
* class.c (rb_class_local_methods): a method to list local methods.
* object.c (Init_Object): add BasicObject class as a top level
BlankSlate class.
* ruby.h (SYM2ID): should not cast to signed long.
[ruby-core:07414]
* class.c (rb_include_module): allow module duplication.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@10235 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2006-06-10 01:20:17 +04:00
}
2010-10-26 21:27:21 +04:00
2020-10-21 16:54:39 +03:00
if ( ! rb_ractor_main_p ( ) & & ! rb_ractor_shareable_p ( val ) ) {
2020-12-21 12:06:28 +03:00
rb_raise ( rb_eRactorIsolationError , " can not set constants with non-shareable objects by non-main Ractors " ) ;
2020-03-09 20:22:11 +03:00
}
2010-10-26 21:27:21 +04:00
check_before_mod_set ( klass , id , val , " constant " ) ;
2020-12-19 21:23:41 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
struct rb_id_table * tbl = RCLASS_CONST_TBL ( klass ) ;
if ( ! tbl ) {
RCLASS_CONST_TBL ( klass ) = tbl = rb_id_table_create ( 0 ) ;
2022-03-31 18:04:25 +03:00
rb_clear_constant_cache_for_id ( id ) ;
2020-12-19 21:23:41 +03:00
ce = ZALLOC ( rb_const_entry_t ) ;
rb_id_table_insert ( tbl , id , ( VALUE ) ce ) ;
setup_const_entry ( ce , klass , val , CONST_PUBLIC ) ;
}
else {
struct autoload_const ac = {
2022-05-16 15:50:02 +03:00
. module = klass , . name = id ,
2020-12-19 21:23:41 +03:00
. value = val , . flag = CONST_PUBLIC ,
/* fill the rest with 0 */
} ;
2022-03-12 08:43:47 +03:00
ac . file = rb_source_location ( & ac . line ) ;
2022-05-15 07:07:12 +03:00
const_tbl_update ( & ac , false ) ;
2020-12-19 21:23:41 +03:00
}
2015-04-13 10:54:59 +03:00
}
2020-12-19 21:23:41 +03:00
RB_VM_LOCK_LEAVE ( ) ;
resolve class name earlier and more consistently
This further avoids class name resolution issues which came
about due to relying on hash table ordering before r53376.
Pre-caching the class name when it is never used raises memory
use, but the overall gain from moving away from st still gives
us a small gain. Reverting r53376 and this patch and testing with
"valgrind -v ./ruby -rrdoc -eexit" on x86 (32-bit) shows:
before:
in use at exit: 1,662,239 bytes in 25,286 blocks
total heap usage: 49,514 allocs, 24,228 frees, 6,005,561 bytes allocated
after, with this change:
in use at exit: 1,646,529 bytes in 24,572 blocks
total heap usage: 48,891 allocs, 24,319 frees, 6,003,921 bytes allocated
* class.c (Init_class_hierarchy): resolve name for rb_cObject ASAP
* object.c (rb_mod_const_set): move name resolution to rb_const_set
* variable.c (rb_const_set): do class resolution here
[ruby-core:72807] [Bug #11977]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53518 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2016-01-12 22:26:07 +03:00
/*
* Resolve and cache class name immediately to resolve ambiguity
* and avoid order - dependency on const_tbl
*/
2019-03-09 03:12:52 +03:00
if ( rb_cObject & & rb_namespace_p ( val ) ) {
2023-01-11 17:23:03 +03:00
bool val_path_permanent ;
2019-04-12 06:46:28 +03:00
VALUE val_path = classname ( val , & val_path_permanent ) ;
if ( NIL_P ( val_path ) | | ! val_path_permanent ) {
2022-04-08 10:44:03 +03:00
if ( klass = = rb_cObject ) {
2019-04-19 00:37:01 +03:00
set_namespace_path ( val , rb_id2str ( id ) ) ;
2022-04-08 10:44:03 +03:00
}
else {
2023-01-11 17:23:03 +03:00
bool parental_path_permanent ;
2019-04-12 06:46:28 +03:00
VALUE parental_path = classname ( klass , & parental_path_permanent ) ;
2020-09-02 04:22:20 +03:00
if ( NIL_P ( parental_path ) ) {
2023-01-11 17:23:03 +03:00
bool throwaway ;
2020-09-02 20:58:29 +03:00
parental_path = rb_tmp_class_path ( klass , & throwaway , make_temporary_path ) ;
2020-09-02 04:22:20 +03:00
}
if ( parental_path_permanent & & ! val_path_permanent ) {
set_namespace_path ( val , build_const_path ( parental_path , id ) ) ;
}
else if ( ! parental_path_permanent & & NIL_P ( val_path ) ) {
2023-01-11 17:23:03 +03:00
RCLASS_SET_CLASSPATH ( val , build_const_path ( parental_path , id ) , false ) ;
2020-09-02 04:22:20 +03:00
}
2022-04-08 10:44:03 +03:00
}
}
resolve class name earlier and more consistently
This further avoids class name resolution issues which came
about due to relying on hash table ordering before r53376.
Pre-caching the class name when it is never used raises memory
use, but the overall gain from moving away from st still gives
us a small gain. Reverting r53376 and this patch and testing with
"valgrind -v ./ruby -rrdoc -eexit" on x86 (32-bit) shows:
before:
in use at exit: 1,662,239 bytes in 25,286 blocks
total heap usage: 49,514 allocs, 24,228 frees, 6,005,561 bytes allocated
after, with this change:
in use at exit: 1,646,529 bytes in 24,572 blocks
total heap usage: 48,891 allocs, 24,319 frees, 6,003,921 bytes allocated
* class.c (Init_class_hierarchy): resolve name for rb_cObject ASAP
* object.c (rb_mod_const_set): move name resolution to rb_const_set
* variable.c (rb_const_set): do class resolution here
[ruby-core:72807] [Bug #11977]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53518 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2016-01-12 22:26:07 +03:00
}
2022-04-08 10:44:03 +03:00
}
void
rb_const_set ( VALUE klass , ID id , VALUE val )
{
const_set ( klass , id , val ) ;
2021-05-22 13:04:01 +03:00
const_added ( klass , id ) ;
2015-04-13 10:54:59 +03:00
}
2022-05-16 15:50:02 +03:00
static struct autoload_data *
2022-05-25 14:12:54 +03:00
autoload_data_for_named_constant ( VALUE module , ID name , struct autoload_const * * autoload_const_pointer )
2018-02-13 12:36:35 +03:00
{
2022-05-25 14:12:54 +03:00
VALUE autoload_data_value = autoload_data ( module , name ) ;
if ( ! autoload_data_value ) return 0 ;
struct autoload_data * autoload_data = get_autoload_data ( autoload_data_value , autoload_const_pointer ) ;
if ( ! autoload_data ) return 0 ;
2022-05-08 01:22:58 +03:00
2018-02-13 12:36:35 +03:00
/* for autoloading thread, keep the defined value to autoloading storage */
2022-05-25 14:12:54 +03:00
if ( autoload_by_current ( autoload_data ) ) {
return autoload_data ;
2018-02-13 12:36:35 +03:00
}
2022-05-25 14:12:54 +03:00
2018-02-13 12:36:35 +03:00
return 0 ;
}
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
static void
2022-05-15 07:07:12 +03:00
const_tbl_update ( struct autoload_const * ac , int autoload_force )
2015-04-13 10:54:59 +03:00
{
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
VALUE value ;
2022-05-16 15:50:02 +03:00
VALUE klass = ac - > module ;
2018-05-10 11:54:26 +03:00
VALUE val = ac - > value ;
2022-05-16 15:50:02 +03:00
ID id = ac - > name ;
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
struct rb_id_table * tbl = RCLASS_CONST_TBL ( klass ) ;
2018-05-10 11:54:26 +03:00
rb_const_flag_t visibility = ac - > flag ;
2015-04-13 10:54:59 +03:00
rb_const_entry_t * ce ;
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
if ( rb_id_table_lookup ( tbl , id , & value ) ) {
2022-05-08 01:22:58 +03:00
ce = ( rb_const_entry_t * ) value ;
2022-11-15 07:24:08 +03:00
if ( UNDEF_P ( ce - > value ) ) {
2022-05-16 15:50:02 +03:00
RUBY_ASSERT_CRITICAL_SECTION_ENTER ( ) ;
2022-03-12 08:43:47 +03:00
VALUE file = ac - > file ;
int line = ac - > line ;
2022-05-16 15:50:02 +03:00
struct autoload_data * ele = autoload_data_for_named_constant ( klass , id , & ac ) ;
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
2022-05-15 07:07:12 +03:00
if ( ! autoload_force & & ele ) {
2022-05-08 01:22:58 +03:00
rb_clear_constant_cache_for_id ( id ) ;
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
2022-05-16 15:50:02 +03:00
ac - > value = val ; /* autoload_data is non-WB-protected */
2020-04-07 15:35:28 +03:00
ac - > file = rb_source_location ( & ac - > line ) ;
2022-05-08 01:22:58 +03:00
}
2020-04-07 15:35:28 +03:00
else {
/* otherwise autoloaded constant, allow to override */
autoload_delete ( klass , id ) ;
ce - > flag = visibility ;
RB_OBJ_WRITE ( klass , & ce - > value , val ) ;
2022-03-12 08:43:47 +03:00
RB_OBJ_WRITE ( klass , & ce - > file , file ) ;
ce - > line = line ;
2020-04-07 15:35:28 +03:00
}
2022-05-16 15:50:02 +03:00
RUBY_ASSERT_CRITICAL_SECTION_LEAVE ( ) ;
2020-04-07 15:35:28 +03:00
return ;
2022-05-08 01:22:58 +03:00
}
else {
VALUE name = QUOTE_ID ( id ) ;
visibility = ce - > flag ;
if ( klass = = rb_cObject )
rb_warn ( " already initialized constant % " PRIsVALUE " " , name ) ;
else
rb_warn ( " already initialized constant % " PRIsVALUE " ::% " PRIsVALUE " " ,
rb_class_name ( klass ) , name ) ;
if ( ! NIL_P ( ce - > file ) & & ce - > line ) {
rb_compile_warn ( RSTRING_PTR ( ce - > file ) , ce - > line ,
" previous definition of % " PRIsVALUE " was here " , name ) ;
}
}
rb_clear_constant_cache_for_id ( id ) ;
setup_const_entry ( ce , klass , val , visibility ) ;
2017-01-08 04:59:54 +03:00
}
else {
2022-05-08 01:22:58 +03:00
rb_clear_constant_cache_for_id ( id ) ;
* class.c, compile.c, eval.c, gc.h, insns.def, internal.h, method.h,
variable.c, vm.c, vm_core.c, vm_insnhelper.c, vm_insnhelper.h,
vm_method.c: Implement class hierarchy method cache invalidation.
[ruby-core:55053] [Feature #8426] [GH-387]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@42822 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2013-09-04 09:25:06 +04:00
2022-05-08 01:22:58 +03:00
ce = ZALLOC ( rb_const_entry_t ) ;
rb_id_table_insert ( tbl , id , ( VALUE ) ce ) ;
setup_const_entry ( ce , klass , val , visibility ) ;
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
}
2015-04-13 10:54:52 +03:00
}
static void
2015-08-04 06:13:19 +03:00
setup_const_entry ( rb_const_entry_t * ce , VALUE klass , VALUE val ,
rb_const_flag_t visibility )
2015-04-13 10:54:52 +03:00
{
2013-06-07 17:25:41 +04:00
ce - > flag = visibility ;
* include/ruby/ruby.h: rename OBJ_WRITE and OBJ_WRITTEN into
RB_OBJ_WRITE and RB_OBJ_WRITTEN.
* array.c, class.c, compile.c, hash.c, internal.h, iseq.c,
proc.c, process.c, re.c, string.c, variable.c, vm.c,
vm_eval.c, vm_insnhelper.c, vm_insnhelper.h,
vm_method.c: catch up this change.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@44299 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2013-12-20 12:07:47 +04:00
RB_OBJ_WRITE ( klass , & ce - > value , val ) ;
2015-10-31 04:02:26 +03:00
RB_OBJ_WRITE ( klass , & ce - > file , rb_source_location ( & ce - > line ) ) ;
2000-02-18 09:59:36 +03:00
}
1998-01-16 15:13:05 +03:00
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_define_const ( VALUE klass , const char * name , VALUE val )
1998-01-16 15:13:05 +03:00
{
ID id = rb_intern ( name ) ;
1999-01-20 07:59:39 +03:00
2003-07-24 09:18:47 +04:00
if ( ! rb_is_const_id ( id ) ) {
2024-01-19 10:03:38 +03:00
rb_warn ( " rb_define_const: invalid name '%s' for constant " , name ) ;
2003-07-24 09:18:47 +04:00
}
2024-03-03 12:46:46 +03:00
if ( ! RB_SPECIAL_CONST_P ( val ) ) {
rb_vm_register_global_object ( val ) ;
}
1999-12-14 09:50:43 +03:00
rb_const_set ( klass , id , val ) ;
1998-01-16 15:13:05 +03:00
}
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_define_global_const ( const char * name , VALUE val )
1998-01-16 15:13:05 +03:00
{
1999-01-20 07:59:39 +03:00
rb_define_const ( rb_cObject , name , val ) ;
1998-01-16 15:13:05 +03:00
}
2010-10-26 21:27:44 +04:00
static void
2015-08-04 06:13:19 +03:00
set_const_visibility ( VALUE mod , int argc , const VALUE * argv ,
rb_const_flag_t flag , rb_const_flag_t mask )
2010-10-26 21:27:44 +04:00
{
int i ;
2014-08-04 05:12:53 +04:00
rb_const_entry_t * ce ;
2010-10-26 21:27:44 +04:00
ID id ;
2018-07-27 16:57:14 +03:00
rb_class_modify_check ( mod ) ;
2011-12-03 15:52:14 +04:00
if ( argc = = 0 ) {
2012-12-22 19:04:57 +04:00
rb_warning ( " % " PRIsVALUE " with no argument is just ignored " ,
2013-05-25 06:24:33 +04:00
QUOTE_ID ( rb_frame_callee ( ) ) ) ;
2013-05-25 06:28:51 +04:00
return ;
2011-12-03 15:52:14 +04:00
}
2010-10-26 21:27:44 +04:00
for ( i = 0 ; i < argc ; i + + ) {
2018-05-10 11:54:26 +03:00
struct autoload_const * ac ;
2011-07-26 20:05:35 +04:00
VALUE val = argv [ i ] ;
id = rb_check_id ( & val ) ;
if ( ! id ) {
2018-12-28 13:33:02 +03:00
undefined_constant ( mod , val ) ;
2011-07-26 20:05:35 +04:00
}
2014-08-04 05:12:53 +04:00
if ( ( ce = rb_const_lookup ( mod , id ) ) ) {
2015-07-30 07:20:00 +03:00
ce - > flag & = ~ mask ;
ce - > flag | = flag ;
2022-11-15 07:24:08 +03:00
if ( UNDEF_P ( ce - > value ) ) {
2022-05-16 15:50:02 +03:00
struct autoload_data * ele ;
2022-07-21 19:23:58 +03:00
2022-05-16 15:50:02 +03:00
ele = autoload_data_for_named_constant ( mod , id , & ac ) ;
2018-02-13 12:36:35 +03:00
if ( ele ) {
2018-05-10 11:54:26 +03:00
ac - > flag & = ~ mask ;
ac - > flag | = flag ;
2018-02-13 12:36:35 +03:00
}
}
2022-03-31 18:04:25 +03:00
rb_clear_constant_cache_for_id ( id ) ;
2010-10-26 21:27:44 +04:00
}
2011-12-08 18:47:19 +04:00
else {
2018-12-28 13:33:02 +03:00
undefined_constant ( mod , ID2SYM ( id ) ) ;
2011-12-08 18:47:19 +04:00
}
2010-10-26 21:27:44 +04:00
}
}
2016-09-09 06:02:22 +03:00
void
rb_deprecate_constant ( VALUE mod , const char * name )
{
rb_const_entry_t * ce ;
ID id ;
long len = strlen ( name ) ;
2018-07-27 16:57:14 +03:00
rb_class_modify_check ( mod ) ;
2018-12-28 13:33:02 +03:00
if ( ! ( id = rb_check_id_cstr ( name , len , NULL ) ) ) {
2018-12-28 13:33:04 +03:00
undefined_constant ( mod , rb_fstring_new ( name , len ) ) ;
2018-12-28 13:33:02 +03:00
}
if ( ! ( ce = rb_const_lookup ( mod , id ) ) ) {
undefined_constant ( mod , ID2SYM ( id ) ) ;
2016-09-09 06:02:22 +03:00
}
ce - > flag | = CONST_DEPRECATED ;
}
2010-10-26 21:27:44 +04:00
/*
* call - seq :
* mod . private_constant ( symbol , . . . ) = > mod
*
* Makes a list of existing constants private .
*/
VALUE
2014-06-18 10:16:39 +04:00
rb_mod_private_constant ( int argc , const VALUE * argv , VALUE obj )
2010-10-26 21:27:44 +04:00
{
2015-07-30 07:20:00 +03:00
set_const_visibility ( obj , argc , argv , CONST_PRIVATE , CONST_VISIBILITY_MASK ) ;
2010-10-26 21:27:44 +04:00
return obj ;
}
/*
* call - seq :
* mod . public_constant ( symbol , . . . ) = > mod
*
* Makes a list of existing constants public .
*/
VALUE
2014-06-18 10:16:39 +04:00
rb_mod_public_constant ( int argc , const VALUE * argv , VALUE obj )
2010-10-26 21:27:44 +04:00
{
2015-07-30 07:20:00 +03:00
set_const_visibility ( obj , argc , argv , CONST_PUBLIC , CONST_VISIBILITY_MASK ) ;
return obj ;
}
2016-01-04 10:08:05 +03:00
/*
* call - seq :
* mod . deprecate_constant ( symbol , . . . ) = > mod
*
2019-12-22 00:05:45 +03:00
* Makes a list of existing constants deprecated . Attempt
* to refer to them will produce a warning .
*
* module HTTP
* NotFound = Exception . new
* NOT_FOUND = NotFound # previous version of the library used this name
*
* deprecate_constant : NOT_FOUND
* end
*
* HTTP : : NOT_FOUND
* # warning : constant HTTP : : NOT_FOUND is deprecated
*
2016-01-04 10:08:05 +03:00
*/
2015-07-30 07:20:00 +03:00
VALUE
rb_mod_deprecate_constant ( int argc , const VALUE * argv , VALUE obj )
{
set_const_visibility ( obj , argc , argv , CONST_DEPRECATED , CONST_DEPRECATED ) ;
2010-10-26 21:27:44 +04:00
return obj ;
}
2007-10-03 10:48:06 +04:00
static VALUE
2007-12-24 12:25:27 +03:00
original_module ( VALUE c )
2007-10-03 10:48:06 +04:00
{
2011-09-29 15:07:45 +04:00
if ( RB_TYPE_P ( c , T_ICLASS ) )
2007-10-03 10:48:06 +04:00
return RBASIC ( c ) - > klass ;
return c ;
}
2012-08-06 19:24:01 +04:00
static int
cvar_lookup_at ( VALUE klass , ID id , st_data_t * v )
{
2022-11-01 00:05:37 +03:00
if ( RB_TYPE_P ( klass , T_ICLASS ) ) {
if ( FL_TEST_RAW ( klass , RICLASS_IS_ORIGIN ) ) {
return 0 ;
2022-11-10 03:11:20 +03:00
}
else {
2022-11-01 00:05:37 +03:00
// check the original module
klass = RBASIC ( klass ) - > klass ;
}
}
VALUE n = rb_ivar_lookup ( klass , id , Qundef ) ;
2022-11-15 07:24:08 +03:00
if ( UNDEF_P ( n ) ) return 0 ;
2022-11-01 00:05:37 +03:00
if ( v ) * v = n ;
return 1 ;
2012-08-06 19:24:01 +04:00
}
static VALUE
cvar_front_klass ( VALUE klass )
{
2024-03-06 19:04:22 +03:00
if ( RCLASS_SINGLETON_P ( klass ) ) {
2023-02-15 12:42:52 +03:00
VALUE obj = RCLASS_ATTACHED_OBJECT ( klass ) ;
2019-03-09 03:12:53 +03:00
if ( rb_namespace_p ( obj ) ) {
2012-08-06 19:24:01 +04:00
return obj ;
}
}
return RCLASS_SUPER ( klass ) ;
}
2018-03-15 10:15:26 +03:00
static void
cvar_overtaken ( VALUE front , VALUE target , ID id )
{
if ( front & & target ! = front ) {
2020-06-11 04:53:25 +03:00
if ( original_module ( front ) ! = original_module ( target ) ) {
2020-03-28 01:08:52 +03:00
rb_raise ( rb_eRuntimeError ,
" class variable % " PRIsVALUE " of % " PRIsVALUE " is overtaken by % " PRIsVALUE " " ,
2018-03-15 10:15:26 +03:00
ID2SYM ( id ) , rb_class_name ( original_module ( front ) ) ,
rb_class_name ( original_module ( target ) ) ) ;
}
if ( BUILTIN_TYPE ( front ) = = T_CLASS ) {
2022-11-01 00:05:37 +03:00
rb_ivar_delete ( front , id , Qundef ) ;
2018-03-15 10:15:26 +03:00
}
}
}
2012-08-06 19:24:01 +04:00
# define CVAR_FOREACH_ANCESTORS(klass, v, r) \
for ( klass = cvar_front_klass ( klass ) ; klass ; klass = RCLASS_SUPER ( klass ) ) { \
if ( cvar_lookup_at ( klass , id , ( v ) ) ) { \
r ; \
} \
}
2007-02-02 12:47:55 +03:00
# define CVAR_LOOKUP(v,r) do {\
2020-03-09 20:22:11 +03:00
CVAR_ACCESSOR_SHOULD_BE_MAIN_RACTOR ( ) ; \
2012-08-06 19:24:01 +04:00
if ( cvar_lookup_at ( klass , id , ( v ) ) ) { r ; } \
CVAR_FOREACH_ANCESTORS ( klass , v , r ) ; \
2007-02-02 12:47:55 +03:00
} while ( 0 )
2023-03-11 04:38:42 +03:00
static VALUE
find_cvar ( VALUE klass , VALUE * front , VALUE * target , ID id )
{
VALUE v = Qundef ;
CVAR_LOOKUP ( & v , {
if ( ! * front ) {
* front = klass ;
}
* target = klass ;
} ) ;
return v ;
}
2021-06-01 20:34:06 +03:00
static void
check_for_cvar_table ( VALUE subclass , VALUE key )
{
2022-11-01 00:05:37 +03:00
// Must not check ivar on ICLASS
if ( ! RB_TYPE_P ( subclass , T_ICLASS ) & & RTEST ( rb_ivar_defined ( subclass , key ) ) ) {
2021-06-01 20:34:06 +03:00
RB_DEBUG_COUNTER_INC ( cvar_class_invalidate ) ;
ruby_vm_global_cvar_state + + ;
return ;
}
rb_class_foreach_subclass ( subclass , check_for_cvar_table , key ) ;
}
2007-11-09 11:14:42 +03:00
void
rb_cvar_set ( VALUE klass , ID id , VALUE val )
{
VALUE tmp , front = 0 , target = 0 ;
tmp = klass ;
CVAR_LOOKUP ( 0 , { if ( ! front ) front = klass ; target = klass ; } ) ;
if ( target ) {
2018-03-15 10:15:26 +03:00
cvar_overtaken ( front , target , id ) ;
2007-11-09 11:14:42 +03:00
}
else {
target = tmp ;
}
2010-10-26 21:27:21 +04:00
2020-06-11 01:44:52 +03:00
if ( RB_TYPE_P ( target , T_ICLASS ) ) {
target = RBASIC ( target ) - > klass ;
}
2010-10-26 21:27:21 +04:00
check_before_mod_set ( target , id , val , " class variable " ) ;
2021-06-01 20:34:06 +03:00
int result = rb_class_ivar_set ( target , id , val ) ;
struct rb_id_table * rb_cvc_tbl = RCLASS_CVC_TBL ( target ) ;
if ( ! rb_cvc_tbl ) {
rb_cvc_tbl = RCLASS_CVC_TBL ( target ) = rb_id_table_create ( 2 ) ;
}
struct rb_cvar_class_tbl_entry * ent ;
2021-06-23 03:39:04 +03:00
VALUE ent_data ;
2021-06-01 20:34:06 +03:00
2021-06-23 03:39:04 +03:00
if ( ! rb_id_table_lookup ( rb_cvc_tbl , id , & ent_data ) ) {
2021-06-01 20:34:06 +03:00
ent = ALLOC ( struct rb_cvar_class_tbl_entry ) ;
ent - > class_value = target ;
ent - > global_cvar_state = GET_GLOBAL_CVAR_STATE ( ) ;
2023-06-02 21:25:19 +03:00
ent - > cref = 0 ;
2021-06-01 20:34:06 +03:00
rb_id_table_insert ( rb_cvc_tbl , id , ( VALUE ) ent ) ;
RB_DEBUG_COUNTER_INC ( cvar_inline_miss ) ;
2021-06-23 03:39:04 +03:00
}
else {
ent = ( void * ) ent_data ;
2021-06-01 20:34:06 +03:00
ent - > global_cvar_state = GET_GLOBAL_CVAR_STATE ( ) ;
}
// Break the cvar cache if this is a new class variable
// and target is a module or a subclass with the same
// cvar in this lookup.
if ( result = = 0 ) {
if ( RB_TYPE_P ( target , T_CLASS ) ) {
if ( RCLASS_SUBCLASSES ( target ) ) {
rb_class_foreach_subclass ( target , check_for_cvar_table , id ) ;
}
}
}
2007-11-09 11:14:42 +03:00
}
2000-02-18 09:59:36 +03:00
VALUE
2021-06-01 20:34:06 +03:00
rb_cvar_find ( VALUE klass , ID id , VALUE * front )
2000-02-18 09:59:36 +03:00
{
2021-06-01 20:34:06 +03:00
VALUE target = 0 ;
VALUE value ;
2000-02-18 09:59:36 +03:00
2021-06-01 20:34:06 +03:00
value = find_cvar ( klass , front , & target , id ) ;
2007-10-03 10:48:06 +04:00
if ( ! target ) {
2015-10-28 09:24:12 +03:00
rb_name_err_raise ( " uninitialized class variable %1$s in %2$s " ,
2021-06-01 20:34:06 +03:00
klass , ID2SYM ( id ) ) ;
2007-10-03 10:48:06 +04:00
}
2021-06-01 20:34:06 +03:00
cvar_overtaken ( * front , target , id ) ;
* compile.c (iseq_build_body), error.c (set_syserr, get_syserr),
(syserr_initialize), gc.c (define_final, rb_gc_copy_finalizer),
(run_final), hash.c (rb_hash_aref, rb_hash_lookup2),
(rb_hash_fetch_m, rb_hash_clear, rb_hash_aset, eql_i),
iseq.c (iseq_load, iseq_data_to_ary), marshal.c (r_symlink),
thread.c (rb_thread_local_aref),
variable.c (generic_ivar_remove, ivar_get, rb_const_get_0),
(rb_cvar_get), vm.c (rb_vm_check_redefinition_opt_method),
vm_insnhelper.c (vm_get_ev_const), vm_method.c (remove_method),
ext/iconv/iconv.c (map_charset): use st_data_t.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@29462 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2010-10-12 18:47:23 +04:00
return ( VALUE ) value ;
2000-02-18 09:59:36 +03:00
}
2021-06-01 20:34:06 +03:00
VALUE
rb_cvar_get ( VALUE klass , ID id )
{
VALUE front = 0 ;
return rb_cvar_find ( klass , id , & front ) ;
}
2000-12-08 10:10:38 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_cvar_defined ( VALUE klass , ID id )
2000-02-18 09:59:36 +03:00
{
2007-02-02 12:47:55 +03:00
if ( ! klass ) return Qfalse ;
2007-10-03 10:48:06 +04:00
CVAR_LOOKUP ( 0 , return Qtrue ) ;
2000-02-18 09:59:36 +03:00
return Qfalse ;
}
2015-10-28 09:24:12 +03:00
static ID
cv_intern ( VALUE klass , const char * name )
2000-03-23 11:37:35 +03:00
{
2001-05-02 08:22:21 +04:00
ID id = rb_intern ( name ) ;
if ( ! rb_is_class_id ( id ) ) {
2015-10-28 09:24:12 +03:00
rb_name_err_raise ( " wrong class variable name %1$s " ,
klass , rb_str_new_cstr ( name ) ) ;
2001-05-02 08:22:21 +04:00
}
2015-10-28 09:24:12 +03:00
return id ;
}
void
rb_cv_set ( VALUE klass , const char * name , VALUE val )
{
ID id = cv_intern ( klass , name ) ;
2007-02-02 12:41:47 +03:00
rb_cvar_set ( klass , id , val ) ;
2000-03-23 11:37:35 +03:00
}
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_cv_get ( VALUE klass , const char * name )
2000-03-23 11:37:35 +03:00
{
2015-10-28 09:24:12 +03:00
ID id = cv_intern ( klass , name ) ;
2001-05-02 08:22:21 +04:00
return rb_cvar_get ( klass , id ) ;
2000-03-23 11:37:35 +03:00
}
void
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_define_class_variable ( VALUE klass , const char * name , VALUE val )
2000-02-18 09:59:36 +03:00
{
2021-06-01 10:32:24 +03:00
rb_cv_set ( klass , name , val ) ;
2000-02-18 09:59:36 +03:00
}
2000-09-15 10:00:30 +04:00
static int
2023-12-06 01:54:37 +03:00
cv_i ( ID key , VALUE v , st_data_t a )
2000-09-15 10:00:30 +04:00
{
2012-07-19 10:41:47 +04:00
st_table * tbl = ( st_table * ) a ;
2012-03-31 02:40:54 +04:00
2000-09-15 10:00:30 +04:00
if ( rb_is_class_id ( key ) ) {
2014-10-08 12:27:54 +04:00
st_update ( tbl , ( st_data_t ) key , cv_i_update , 0 ) ;
2000-09-15 10:00:30 +04:00
}
return ST_CONTINUE ;
}
2012-08-23 11:45:46 +04:00
2012-07-19 10:41:47 +04:00
static void *
mod_cvar_at ( VALUE mod , void * data )
{
st_table * tbl = data ;
if ( ! tbl ) {
tbl = st_init_numtable ( ) ;
}
2022-11-01 00:05:37 +03:00
mod = original_module ( mod ) ;
rb_ivar_foreach ( mod , cv_i , ( st_data_t ) tbl ) ;
2012-07-19 10:41:47 +04:00
return tbl ;
}
static void *
mod_cvar_of ( VALUE mod , void * data )
{
VALUE tmp = mod ;
2024-03-06 19:04:22 +03:00
if ( RCLASS_SINGLETON_P ( mod ) ) {
2023-02-15 12:42:52 +03:00
if ( rb_namespace_p ( RCLASS_ATTACHED_OBJECT ( mod ) ) ) {
2019-08-10 02:44:43 +03:00
data = mod_cvar_at ( tmp , data ) ;
tmp = cvar_front_klass ( tmp ) ;
}
}
2012-07-19 10:41:47 +04:00
for ( ; ; ) {
data = mod_cvar_at ( tmp , data ) ;
tmp = RCLASS_SUPER ( tmp ) ;
if ( ! tmp ) break ;
}
return data ;
}
static int
cv_list_i ( st_data_t key , st_data_t value , VALUE ary )
{
ID sym = ( ID ) key ;
rb_ary_push ( ary , ID2SYM ( sym ) ) ;
return ST_CONTINUE ;
}
static VALUE
cvar_list ( void * data )
{
st_table * tbl = data ;
VALUE ary ;
if ( ! tbl ) return rb_ary_new2 ( 0 ) ;
ary = rb_ary_new2 ( tbl - > num_entries ) ;
st_foreach_safe ( tbl , cv_list_i , ary ) ;
st_free_table ( tbl ) ;
return ary ;
}
2000-09-15 10:00:30 +04:00
2003-12-28 09:33:07 +03:00
/*
* call - seq :
2012-07-19 10:41:47 +04:00
* mod . class_variables ( inherit = true ) - > array
2009-02-15 05:45:31 +03:00
*
2004-06-26 10:17:20 +04:00
* Returns an array of the names of class variables in < i > mod < / i > .
2012-07-19 10:41:47 +04:00
* This includes the names of class variables in any included
* modules , unless the < i > inherit < / i > parameter is set to
* < code > false < / code > .
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
* class One
* @ @ var1 = 1
* end
* class Two < One
* @ @ var2 = 2
* end
2013-12-30 18:39:35 +04:00
* One . class_variables # = > [ : @ @ var1 ]
* Two . class_variables # = > [ : @ @ var2 , : @ @ var1 ]
* Two . class_variables ( false ) # = > [ : @ @ var2 ]
2003-12-28 09:33:07 +03:00
*/
2000-09-15 10:00:30 +04:00
VALUE
2014-06-18 10:16:39 +04:00
rb_mod_class_variables ( int argc , const VALUE * argv , VALUE mod )
2000-09-15 10:00:30 +04:00
{
2019-11-25 09:05:53 +03:00
bool inherit = true ;
2012-07-19 10:41:47 +04:00
st_table * tbl ;
2000-09-15 10:00:30 +04:00
2018-12-06 10:49:24 +03:00
if ( rb_check_arity ( argc , 0 , 1 ) ) inherit = RTEST ( argv [ 0 ] ) ;
if ( inherit ) {
2012-07-19 10:41:47 +04:00
tbl = mod_cvar_of ( mod , 0 ) ;
}
else {
tbl = mod_cvar_at ( mod , 0 ) ;
}
return cvar_list ( tbl ) ;
2000-09-15 10:00:30 +04:00
}
2003-12-28 09:33:07 +03:00
/*
* call - seq :
2010-05-18 01:07:33 +04:00
* remove_class_variable ( sym ) - > obj
2009-02-15 05:45:31 +03:00
*
2020-08-02 19:56:06 +03:00
* Removes the named class variable from the receiver , returning that
* variable ' s value .
2009-02-15 05:45:31 +03:00
*
2020-08-02 19:56:06 +03:00
* class Example
2003-12-28 09:33:07 +03:00
* @ @ var = 99
2020-08-02 19:56:06 +03:00
* puts remove_class_variable ( : @ @ var )
2008-10-09 17:07:28 +04:00
* p ( defined ? @ @ var )
2003-12-28 09:33:07 +03:00
* end
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
* < em > produces : < / em >
2009-02-15 05:45:31 +03:00
*
2003-12-28 09:33:07 +03:00
* 99
* nil
*/
2000-12-08 10:10:38 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_mod_remove_cvar ( VALUE mod , VALUE name )
2000-12-08 10:10:38 +03:00
{
2015-10-28 09:24:12 +03:00
const ID id = id_for_var_message ( mod , name , class , " wrong class variable name %1$s " ) ;
2022-11-01 00:05:37 +03:00
st_data_t val ;
2000-12-08 10:10:38 +03:00
2011-07-23 19:05:03 +04:00
if ( ! id ) {
2020-06-22 04:18:28 +03:00
goto not_defined ;
2000-12-08 10:10:38 +03:00
}
* array.c, gc.c, hash.c, object.c, string.c, struct.c,
transcode.c, variable.c, vm.c, vm_insnhelper.c, vm_method.c:
replace calls to rb_error_frozen() with rb_check_frozen(). a
patch from Run Paint Run Run at [ruby-core:32014]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@29583 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2010-10-24 12:14:05 +04:00
rb_check_frozen ( mod ) ;
2022-11-01 00:05:37 +03:00
val = rb_ivar_delete ( mod , id , Qundef ) ;
2022-11-15 07:24:08 +03:00
if ( ! UNDEF_P ( val ) ) {
2008-10-08 06:23:04 +04:00
return ( VALUE ) val ;
2000-12-08 10:10:38 +03:00
}
if ( rb_cvar_defined ( mod , id ) ) {
2015-10-28 09:24:12 +03:00
rb_name_err_raise ( " cannot remove %1$s for %2$s " , mod , ID2SYM ( id ) ) ;
2000-12-08 10:10:38 +03:00
}
2020-06-22 04:18:28 +03:00
not_defined :
rb_name_err_raise ( " class variable %1$s not defined for %2$s " ,
mod , name ) ;
2020-06-24 10:23:59 +03:00
UNREACHABLE_RETURN ( Qundef ) ;
2000-12-08 10:10:38 +03:00
}
1998-01-16 15:13:05 +03:00
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_iv_get ( VALUE obj , const char * name )
1998-01-16 15:13:05 +03:00
{
2019-07-01 07:56:55 +03:00
ID id = rb_check_id_cstr ( name , strlen ( name ) , rb_usascii_encoding ( ) ) ;
1998-01-16 15:13:05 +03:00
2019-07-01 07:56:55 +03:00
if ( ! id ) {
2019-07-01 08:00:37 +03:00
return Qnil ;
2019-07-01 07:56:55 +03:00
}
1998-01-16 15:13:05 +03:00
return rb_ivar_get ( obj , id ) ;
}
VALUE
* array.c: moved to ANSI function style from K&R function style.
(used protoize on windows, so still K&R remains on #ifdef part of
other platforms. And `foo _((boo))' stuff is still there)
[ruby-dev:26975]
* bignum.c, class.c, compar.c, dir.c, dln.c, dmyext.c, enum.c,
enumerator.c, error.c, eval.c, file.c, gc.c, hash.c, inits.c,
io.c, main.c, marshal.c, math.c, numeric.c, object.c, pack.c,
prec.c, process.c, random.c, range.c, re.c, regcomp.c, regenc.c,
regerror.c, regexec.c, regparse.c, regparse.h, ruby.c, signal.c,
sprintf.c, st.c, string.c, struct.c, time.c, util.h, variable.c,
version.c: ditto.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9126 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-09-12 14:44:21 +04:00
rb_iv_set ( VALUE obj , const char * name , VALUE val )
1998-01-16 15:13:05 +03:00
{
ID id = rb_intern ( name ) ;
return rb_ivar_set ( obj , id , val ) ;
}
2013-06-14 13:23:54 +04:00
2023-11-06 16:32:32 +03:00
static VALUE *
class_ivar_set_shape_ivptr ( VALUE obj , void * _data )
2013-06-14 13:23:54 +04:00
{
2023-11-06 16:32:32 +03:00
RUBY_ASSERT ( ! rb_shape_obj_too_complex ( obj ) ) ;
2023-11-02 18:01:35 +03:00
2023-11-06 16:32:32 +03:00
return RCLASS_IVPTR ( obj ) ;
}
2023-10-19 21:00:54 +03:00
2023-11-06 16:32:32 +03:00
static void
class_ivar_set_shape_resize_ivptr ( VALUE obj , attr_index_t _old_capa , attr_index_t new_capa , void * _data )
{
REALLOC_N ( RCLASS_IVPTR ( obj ) , VALUE , new_capa ) ;
}
2022-11-01 00:05:37 +03:00
2023-11-06 16:32:32 +03:00
static void
class_ivar_set_set_shape ( VALUE obj , rb_shape_t * shape , void * _data )
{
rb_shape_set_shape ( obj , shape ) ;
}
2022-11-01 00:05:37 +03:00
2023-11-06 16:32:32 +03:00
static void
class_ivar_set_transition_too_complex ( VALUE obj , void * _data )
{
2023-11-16 18:28:12 +03:00
rb_evict_ivars_to_hash ( obj ) ;
2023-11-06 16:32:32 +03:00
}
2023-11-02 18:01:35 +03:00
2023-11-06 16:32:32 +03:00
static st_table *
class_ivar_set_too_complex_table ( VALUE obj , void * _data )
{
RUBY_ASSERT ( rb_shape_obj_too_complex ( obj ) ) ;
2023-11-02 18:01:35 +03:00
2023-11-06 16:32:32 +03:00
return RCLASS_IV_HASH ( obj ) ;
}
2023-11-02 18:01:35 +03:00
2023-11-06 16:32:32 +03:00
int
rb_class_ivar_set ( VALUE obj , ID id , VALUE val )
{
RUBY_ASSERT ( RB_TYPE_P ( obj , T_CLASS ) | | RB_TYPE_P ( obj , T_MODULE ) ) ;
bool existing = false ;
rb_check_frozen ( obj ) ;
2023-11-02 18:01:35 +03:00
2023-11-06 16:32:32 +03:00
RB_VM_LOCK_ENTER ( ) ;
{
existing = general_ivar_set ( obj , id , val , NULL ,
class_ivar_set_shape_ivptr ,
class_ivar_set_shape_resize_ivptr ,
class_ivar_set_set_shape ,
class_ivar_set_transition_too_complex ,
class_ivar_set_too_complex_table ) . existing ;
2023-11-10 20:43:06 +03:00
}
2022-11-01 00:05:37 +03:00
RB_VM_LOCK_LEAVE ( ) ;
2021-03-10 18:24:28 +03:00
2023-11-06 16:32:32 +03:00
return existing ;
2013-06-14 13:23:54 +04:00
}
static int
2023-12-06 01:54:37 +03:00
tbl_copy_i ( ID key , VALUE val , st_data_t dest )
2022-12-02 17:43:53 +03:00
{
2023-12-06 01:54:37 +03:00
rb_class_ivar_set ( ( VALUE ) dest , key , val ) ;
2022-11-01 00:05:37 +03:00
2013-06-14 13:23:54 +04:00
return ST_CONTINUE ;
}
2019-09-22 10:21:26 +03:00
void
rb_iv_tbl_copy ( VALUE dst , VALUE src )
2013-06-14 13:23:54 +04:00
{
2022-11-01 00:05:37 +03:00
RUBY_ASSERT ( rb_type ( dst ) = = rb_type ( src ) ) ;
RUBY_ASSERT ( RB_TYPE_P ( dst , T_CLASS ) | | RB_TYPE_P ( dst , T_MODULE ) ) ;
2023-11-02 16:49:23 +03:00
RUBY_ASSERT ( rb_shape_get_shape ( dst ) - > type = = SHAPE_ROOT ) ;
2022-11-01 00:05:37 +03:00
RUBY_ASSERT ( ! RCLASS_IVPTR ( dst ) ) ;
rb_ivar_foreach ( src , tbl_copy_i , dst ) ;
2013-06-14 13:23:54 +04:00
}
2014-08-04 05:12:53 +04:00
2023-03-07 08:34:31 +03:00
rb_const_entry_t *
2014-08-04 05:12:53 +04:00
rb_const_lookup ( VALUE klass , ID id )
{
use id_table for constant tables
valgrind 3.9.0 on x86-64 reports a minor reduction in memory usage
when loading only RubyGems and RDoc by running: ruby -rrdoc -eexit
before: HEAP SUMMARY:
in use at exit: 2,913,448 bytes in 27,394 blocks
total heap usage: 48,362 allocs, 20,968 frees, 9,034,621 bytes alloc
after: HEAP SUMMARY:
in use at exit: 2,880,056 bytes in 26,712 blocks
total heap usage: 47,791 allocs, 21,079 frees, 9,046,507 bytes alloc
* class.c (struct clone_const_arg): adjust for id_table
(clone_const): ditto
(clone_const_i): ditto
(rb_mod_init_copy): ditto
(rb_singleton_class_clone_and_attach): ditto
(rb_include_class_new): ditto
(include_modules_at): ditto
* constant.h (rb_free_const_table): ditto
* gc.c (free_const_entry_i): ditto
(rb_free_const_table): ditto
(obj_memsize_of): ditto
(mark_const_entry_i): ditto
(mark_const_tbl): ditto
* internal.h (struct rb_classext_struct): ditto
* object.c (rb_mod_const_set): resolve class name on assignment
* variable.c (const_update): replace with const_tbl_update
(const_tbl_update): new function
(fc_i): adjust for id_table
(find_class_path): ditto
(autoload_const_set): st_update => const_tbl_update
(rb_const_remove): adjust for id_table
(sv_i): ditto
(rb_local_constants_i): ditto
(rb_local_constants): ditto
(rb_mod_const_at): ditto
(rb_mod_const_set): ditto
(rb_const_lookup): ditto
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@53376 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2015-12-29 23:19:14 +03:00
struct rb_id_table * tbl = RCLASS_CONST_TBL ( klass ) ;
2014-08-04 05:12:53 +04:00
2020-12-19 21:23:41 +03:00
if ( tbl ) {
VALUE val ;
bool r ;
RB_VM_LOCK_ENTER ( ) ;
{
r = rb_id_table_lookup ( tbl , id , & val ) ;
}
RB_VM_LOCK_LEAVE ( ) ;
if ( r ) return ( rb_const_entry_t * ) val ;
2014-08-04 05:12:53 +04:00
}
2020-12-19 21:23:41 +03:00
return NULL ;
2014-08-04 05:12:53 +04:00
}