Ruby 3.5.0dev (2025-04-03 revision 1dddc6c78b5f6dc6ae18ee04ebe44abfce3b0433)
version.c (1dddc6c78b5f6dc6ae18ee04ebe44abfce3b0433)
1/**********************************************************************
2
3 version.c -
4
5 $Author$
6 created at: Thu Sep 30 20:08:01 JST 1993
7
8 Copyright (C) 1993-2007 Yukihiro Matsumoto
9
10**********************************************************************/
11
12#include "internal/cmdlineopt.h"
13#include "internal/parse.h"
14#include "internal/gc.h"
15#include "ruby/ruby.h"
16#include "version.h"
17#include "vm_core.h"
18#include "yjit.h"
19#include <stdio.h>
20
21#ifndef EXIT_SUCCESS
22#define EXIT_SUCCESS 0
23#endif
24
25#ifdef RUBY_REVISION
26# if RUBY_PATCHLEVEL == -1
27# ifndef RUBY_BRANCH_NAME
28# define RUBY_BRANCH_NAME "master"
29# endif
30# define RUBY_REVISION_STR " "RUBY_BRANCH_NAME" "RUBY_REVISION
31# else
32# define RUBY_REVISION_STR " revision "RUBY_REVISION
33# endif
34#else
35# define RUBY_REVISION "HEAD"
36# define RUBY_REVISION_STR ""
37#endif
38#if !defined RUBY_RELEASE_DATETIME || RUBY_PATCHLEVEL != -1
39# undef RUBY_RELEASE_DATETIME
40# define RUBY_RELEASE_DATETIME RUBY_RELEASE_DATE
41#endif
42
43#define PRINT(type) puts(ruby_##type)
44#define MKSTR(type) rb_obj_freeze(rb_usascii_str_new_static(ruby_##type, sizeof(ruby_##type)-1))
45#define MKINT(name) INT2FIX(ruby_##name)
46
52#define RUBY_VERSION \
53 STRINGIZE(RUBY_VERSION_MAJOR) "." \
54 STRINGIZE(RUBY_VERSION_MINOR) "." \
55 STRINGIZE(RUBY_VERSION_TEENY) ""
56#ifndef RUBY_FULL_REVISION
57# define RUBY_FULL_REVISION RUBY_REVISION
58#endif
59#ifdef YJIT_SUPPORT
60#define YJIT_DESCRIPTION " +YJIT " STRINGIZE(YJIT_SUPPORT)
61#else
62#define YJIT_DESCRIPTION " +YJIT"
63#endif
64#if USE_MODULAR_GC
65#define GC_DESCRIPTION " +GC"
66#else
67#define GC_DESCRIPTION ""
68#endif
69const char ruby_version[] = RUBY_VERSION;
70const char ruby_revision[] = RUBY_FULL_REVISION;
71const char ruby_release_date[] = RUBY_RELEASE_DATE;
72const char ruby_platform[] = RUBY_PLATFORM;
73const int ruby_patchlevel = RUBY_PATCHLEVEL;
74const char ruby_description[] =
75 "ruby " RUBY_VERSION RUBY_PATCHLEVEL_STR " "
76 "(" RUBY_RELEASE_DATETIME RUBY_REVISION_STR ") "
77 "[" RUBY_PLATFORM "]";
78static const int ruby_description_opt_point =
79 (int)(sizeof(ruby_description) - sizeof(" [" RUBY_PLATFORM "]"));
80
81const char ruby_copyright[] = "ruby - Copyright (C) "
82 RUBY_BIRTH_YEAR_STR "-" RUBY_RELEASE_YEAR_STR " "
84const char ruby_engine[] = "ruby";
85
86// Might change after initialization
87const char *rb_dynamic_description = ruby_description;
88
89static inline void
90define_ruby_const(VALUE mod, const char *name, VALUE value, bool toplevel)
91{
92 if (toplevel) {
93 rb_define_global_const(name, value);
94 name += rb_strlen_lit("RUBY_");
95 }
96 rb_define_const(mod, name, value);
97}
98
99/* RDoc needs rb_define_const */
100#define rb_define_const(mod, name, value) \
101 define_ruby_const(mod, (mod == mRuby ? "RUBY_" name : name), value, (mod == mRuby))
102
104void
105Init_version(void)
106{
107 /*
108 * The Ruby[rdoc-ref:Ruby] module that contains portable information among
109 * implementations.
110 *
111 * The constants defined here are aliased in the toplevel with
112 * +RUBY_+ prefix.
113 */
114 VALUE mRuby = rb_define_module("Ruby");
115
116 enum {ruby_patchlevel = RUBY_PATCHLEVEL};
117 VALUE version = MKSTR(version);
118 VALUE ruby_engine_name = MKSTR(engine);
119 // MKSTR macro is a marker for fake.rb
120
121 /*
122 * The running version of ruby
123 */
124 rb_define_const(mRuby, "VERSION", /* MKSTR(version) */ version);
125 /*
126 * The date this ruby was released
127 */
128 rb_define_const(mRuby, "RELEASE_DATE", MKSTR(release_date));
129 /*
130 * The platform for this ruby
131 */
132 rb_define_const(mRuby, "PLATFORM", MKSTR(platform));
133 /*
134 * The patchlevel for this ruby. If this is a development build of ruby
135 * the patchlevel will be -1
136 */
137 rb_define_const(mRuby, "PATCHLEVEL", MKINT(patchlevel));
138 /*
139 * The GIT commit hash for this ruby.
140 */
141 rb_define_const(mRuby, "REVISION", MKSTR(revision));
142 /*
143 * The copyright string for ruby
144 */
145 rb_define_const(mRuby, "COPYRIGHT", MKSTR(copyright));
146 /*
147 * The engine or interpreter this ruby uses.
148 */
149 rb_define_const(mRuby, "ENGINE", /* MKSTR(engine) */ ruby_engine_name);
150 ruby_set_script_name(ruby_engine_name);
151 /*
152 * The version of the engine or interpreter this ruby uses.
153 */
154 rb_define_const(mRuby, "ENGINE_VERSION", /* MKSTR(version) */ version);
155
156 rb_provide("ruby2_keywords.rb");
157}
158
159#if USE_YJIT
160#define YJIT_OPTS_ON opt->yjit
161#else
162#define YJIT_OPTS_ON 0
163#endif
164
165int ruby_mn_threads_enabled;
166
167#ifndef RB_DEFAULT_PARSER
168#define RB_DEFAULT_PARSER RB_DEFAULT_PARSER_PRISM
169#endif
170static ruby_default_parser_enum default_parser = RB_DEFAULT_PARSER;
171
172ruby_default_parser_enum
173rb_ruby_default_parser(void)
174{
175 return default_parser;
176}
177
178void
179rb_ruby_default_parser_set(ruby_default_parser_enum parser)
180{
181 default_parser = parser;
182}
183
184static void
185define_ruby_description(const char *const jit_opt)
186{
187 static char desc[
188 sizeof(ruby_description)
189 + rb_strlen_lit(YJIT_DESCRIPTION)
190 + rb_strlen_lit(" +MN")
191 + rb_strlen_lit(" +PRISM")
192#if USE_MODULAR_GC
193 + rb_strlen_lit(GC_DESCRIPTION)
194 // Assume the active GC name can not be longer than 20 chars
195 // so that we don't have to use strlen and remove the static
196 // qualifier from desc.
197 + RB_GC_MAX_NAME_LEN + 3
198#endif
199
200 ];
201
202 int n = ruby_description_opt_point;
203 memcpy(desc, ruby_description, n);
204# define append(s) (n += (int)strlcpy(desc + n, s, sizeof(desc) - n))
205 if (*jit_opt) append(jit_opt);
206 RUBY_ASSERT(n <= ruby_description_opt_point + (int)rb_strlen_lit(YJIT_DESCRIPTION));
207 if (ruby_mn_threads_enabled) append(" +MN");
208 if (rb_ruby_prism_p()) append(" +PRISM");
209#if USE_MODULAR_GC
210 append(GC_DESCRIPTION);
211 if (rb_gc_modular_gc_loaded_p()) {
212 append("[");
213 append(rb_gc_active_gc_name());
214 append("]");
215 }
216#endif
217 append(ruby_description + ruby_description_opt_point);
218# undef append
219
220 VALUE mRuby = rb_path2class("Ruby");
221 VALUE description = rb_obj_freeze(rb_usascii_str_new_static(desc, n));
222 rb_dynamic_description = desc;
223
224 /*
225 * The full ruby version string, like <tt>ruby -v</tt> prints
226 */
227 rb_define_const(mRuby, "DESCRIPTION", /* MKSTR(description) */ description);
228}
229
230void
231Init_ruby_description(ruby_cmdline_options_t *opt)
232{
233 const char *const jit_opt =
234 YJIT_OPTS_ON ? YJIT_DESCRIPTION :
235 "";
236 define_ruby_description(jit_opt);
237}
238
239void
240ruby_set_yjit_description(void)
241{
242 VALUE mRuby = rb_path2class("Ruby");
243 rb_const_remove(rb_cObject, rb_intern("RUBY_DESCRIPTION"));
244 rb_const_remove(mRuby, rb_intern("DESCRIPTION"));
245 define_ruby_description(YJIT_DESCRIPTION);
246}
247
248void
250{
251 puts(rb_dynamic_description);
252
253#ifdef RUBY_LAST_COMMIT_TITLE
254 fputs("last_commit=" RUBY_LAST_COMMIT_TITLE, stdout);
255#endif
256#ifdef HAVE_MALLOC_CONF
257 if (malloc_conf) printf("malloc_conf=%s\n", malloc_conf);
258#endif
259 fflush(stdout);
260}
261
262void
264{
265 PRINT(copyright);
266 fflush(stdout);
267}
#define RUBY_ASSERT(...)
Asserts that the given expression is truthy if and only if RUBY_DEBUG is truthy.
Definition assert.h:219
VALUE rb_define_module(const char *name)
Defines a top-level module.
Definition class.c:1098
void ruby_set_script_name(VALUE name)
Identical to ruby_script(), except it takes the name as a Ruby String instance.
Definition ruby.c:3000
VALUE rb_obj_freeze(VALUE obj)
Just calls rb_obj_freeze_inline() inside.
Definition object.c:1260
void ruby_show_copyright(void)
Prints the copyright notice of the CRuby interpreter to stdout.
Definition version.c:263
void ruby_show_version(void)
Prints the version information of the CRuby interpreter to stdout.
Definition version.c:249
void rb_provide(const char *feature)
Declares that the given feature is already provided by someone else.
Definition load.c:715
#define rb_strlen_lit(str)
Length of a string literal.
Definition string.h:1692
VALUE rb_usascii_str_new_static(const char *ptr, long len)
Identical to rb_str_new_static(), except it generates a string of "US ASCII" encoding instead of "bin...
Definition string.c:1147
VALUE rb_path2class(const char *path)
Resolves a Q::W::E::R-style path string to the actual class it points.
Definition variable.c:486
VALUE rb_const_remove(VALUE space, ID name)
Identical to rb_mod_remove_const(), except it takes the name as ID instead of VALUE.
Definition variable.c:3309
void rb_define_global_const(const char *name, VALUE val)
Identical to rb_define_const(), except it defines that of "global", i.e.
Definition variable.c:3776
const int ruby_api_version[3]
API versions, in { major, minor, teeny } order.
Definition version.c:47
const char ruby_engine[]
This is just "ruby" for us.
Definition version.c:84
#define RUBY_API_VERSION_TEENY
Teeny version.
Definition version.h:76
const char ruby_platform[]
Target platform identifier, in a C string.
Definition version.c:72
const char ruby_version[]
Stringised version.
Definition version.c:69
#define RUBY_API_VERSION_MAJOR
Major version.
Definition version.h:64
#define RUBY_API_VERSION_MINOR
Minor version.
Definition version.h:70
#define RUBY_AUTHOR
Author of this project.
Definition version.h:32
const char ruby_copyright[]
Copyright notice.
Definition version.c:81
const char ruby_release_date[]
Date of release, in a C string.
Definition version.c:71
const int ruby_patchlevel
This is a monotonic increasing integer that describes specific "patch" level.
Definition version.c:73
const char ruby_description[]
This is what ruby -v prints to the standard error.
Definition version.c:74
uintptr_t VALUE
Type that represents a Ruby object.
Definition value.h:40