2 This file is part of LilyPond, the GNU music typesetter.
4 Copyright (C) 2005--2014 Han-Wen Nienhuys <hanwen@xs4all.nl>
6 LilyPond is free software: you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation, either version 3 of the License, or
9 (at your option) any later version.
11 LilyPond is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License
17 along with LilyPond. If not, see <http://www.gnu.org/licenses/>.
23 // Contains generic template definitions. With GCC, it is just
24 // included from smobs.hh, but other template expansion systems might
25 // make it feasible to compile this only a single time.
27 #include "lily-guile-macros.hh"
31 template <class Super>
33 Smob_base<Super>::mark_trampoline (SCM arg)
35 return (Super::unsmob (arg))->mark_smob ();
38 template <class Super>
40 Smob_base<Super>::print_trampoline (SCM arg, SCM port, scm_print_state *p)
42 return (Super::unsmob (arg))->print_smob (port, p);
45 template <class Super>
47 Smob_base<Super>::register_ptr (Super *p)
49 // Don't use SCM_RETURN_NEWSMOB since that would require us to
50 // first register the memory and then create the smob. That would
51 // announce the memory as being GC-controlled before even
52 // allocating the controlling smob.
53 SCM s = SCM_UNDEFINED;
54 SCM_NEWSMOB (s, smob_tag (), p);
55 scm_gc_register_collectable_memory (p, sizeof (*p), smob_name_.c_str ());
59 // Defaults, should not actually get called
60 template <class Super>
62 Smob_base<Super>::mark_smob ()
64 return SCM_UNSPECIFIED;
67 template <class Super>
69 Smob_base<Super>::free_smob (SCM)
74 template <class Super>
76 Smob_base<Super>::equal_p (SCM, SCM)
81 // Default, will often get called
83 template <class Super>
85 Smob_base<Super>::print_smob (SCM p, scm_print_state *)
88 scm_puts (smob_name_.c_str (), p);
93 template <class Super>
95 Smob_base<Super>::unregister_ptr (SCM obj)
97 Super *p = Super::unchecked_unsmob (obj);
98 scm_gc_unregister_collectable_memory (p, sizeof (*p), smob_name_.c_str ());
102 template <class Super>
103 scm_t_bits Smob_base<Super>::smob_tag_ = 0;
105 template <class Super>
106 Scm_init Smob_base<Super>::scm_init_ = init;
108 template <class Super>
109 string Smob_base<Super>::smob_name_;
111 template <class Super>
112 void Smob_base<Super>::init ()
114 smob_name_ = typeid (Super).name ();
115 // Primitive demangling, suitable for GCC, should be harmless
116 // elsewhere. The worst that can happen is that we get material
117 // unsuitable for Texinfo documentation. If that proves to be an
118 // issue, we need some smarter strategy.
119 smob_name_ = smob_name_.substr (smob_name_.find_first_not_of ("0123456789"));
121 smob_tag_ = scm_make_smob_type (smob_name_.c_str (), 0);
122 // The following have trivial private default definitions not
123 // referring to any aspect of the Super class apart from its name.
124 // They should be overridden (or rather masked) at Super level: that
125 // way they can refer to Super-private data.
126 // While that's not a consideration for type_p_name_, it's easier
127 // doing it like the rest.
129 if (&Super::free_smob != &Smob_base<Super>::free_smob)
130 scm_set_smob_free (smob_tag_, Super::free_smob);
131 // Old GCC versions refuse comparing pointers-to-member-function of
132 // covariant types, so we recast here.
133 if (&Super::mark_smob !=
134 static_cast<SCM (Super::*)()>(&Smob_base<Super>::mark_smob))
135 scm_set_smob_mark (smob_tag_, Super::mark_trampoline);
136 scm_set_smob_print (smob_tag_, Super::print_trampoline);
137 if (&Super::equal_p != &Smob_base<Super>::equal_p)
138 scm_set_smob_equalp (smob_tag_, Super::equal_p);
139 if (Super::type_p_name_ != 0)
141 SCM subr = scm_c_define_gsubr (Super::type_p_name_, 1, 0, 0,
142 (scm_t_subr) smob_p);
143 string fundoc = string("Is @var{x} a @code{") + smob_name_
145 ly_add_function_documentation (subr, Super::type_p_name_, "(SCM x)",
147 scm_c_export (Super::type_p_name_, NULL);
149 ly_add_type_predicate ((void *) is_smob, smob_name_.c_str ());
150 if (Super::smob_proc_signature_ >= 0)
151 scm_set_smob_apply (smob_tag_,
152 (scm_t_subr)Super::smob_proc,
153 Super::smob_proc_signature_ >> 8,
154 (Super::smob_proc_signature_ >> 4)&0xf,
155 Super::smob_proc_signature_ & 0xf);