Skip to content

Commit 13816db

Browse files
briotCode Review
authored and
Code Review
committed
Add support for overriding virtual-method for classes
This was only possible for interfaces, but in fact for a GApplication we might need to override the local_command_line virtual_method, as per the documentation. Group all with statements for which we need a pragma Warnings(Off) so that a single instance of the pragma is needed. Minor style fixes N625-006 Change-Id: I36aa3285fbabc964e198f1b1e2f0cf21a2210037
1 parent a78175a commit 13816db

12 files changed

+120
-52
lines changed

contrib/adaformat.py

+38-26
Original file line numberDiff line numberDiff line change
@@ -85,10 +85,12 @@ def __init__(self, ada, property):
8585
self.is_ptr = False
8686
self.param = ada # type as parameter
8787
self.cparam = ada # type for Ada subprograms binding to C
88-
self.cleanup = None # If set, a tmp variable is created to hold the
89-
# result of convert during the call, and is then
90-
# free by calling this cleanup. Use "%s" as the
91-
# name of the variable.
88+
89+
self.cleanup = None
90+
# If set, a tmp variable is created to hold the result of convert during
91+
# the call, and is then free by calling this cleanup. Use "%s" as the
92+
# name of the variable.
93+
9294
self.isArray = False
9395

9496
# In some cases, Ada provides a special value for a parameter that
@@ -132,8 +134,8 @@ def convert_to_c(self, pkg=None):
132134
It can also use %(var)s which will be substituted by the name of the
133135
parameter.
134136
Otherwise, it is used as " Tmp := <convert>".
135-
It might be necessary to also override add_with() to add the necessary
136-
with statements.
137+
It might be necessary to also override add_with() to add the
138+
necessary with statements.
137139
"""
138140

139141
if self.allow_none and self.val_or_null:
@@ -193,8 +195,8 @@ def as_c_param(self, pkg=None):
193195
return self.cparam
194196

195197
def as_call(
196-
self, name, pkg, wrapper="%s", lang="ada", mode="in", value=None,
197-
is_temporary_variable=True):
198+
self, name, pkg, wrapper="%s", lang="ada", mode="in", value=None,
199+
is_temporary_variable=True):
198200
"""'name' represents a parameter of type 'self'.
199201
'pkg' is the Package instance in which the call occurs.
200202
'wrapper' is used in the call itself, and %s is replaced by the
@@ -227,18 +229,20 @@ def as_call(
227229
# Unless we are already using a temporary variable.
228230

229231
if (ret
230-
and ret != "%(var)s"
231-
and mode != "in"
232-
and not is_temporary_variable):
232+
and ret != "%(var)s"
233+
and mode != "in"
234+
and not is_temporary_variable):
233235

234236
tmp = "Tmp_%s" % name
235237

236238
if mode in ("out", "access"):
237-
tmpvars = [Local_Var(name=tmp, type=returns[4], aliased=True)]
239+
tmpvars = [Local_Var(
240+
name=tmp, type=returns[4], aliased=True)]
238241
else:
239242
tmpvars = [
240243
Local_Var(name=tmp, type=returns[4], aliased=True,
241-
default=self.convert_to_c(pkg=pkg) % {"var":name})]
244+
default=self.convert_to_c(pkg=pkg) % {
245+
"var": name})]
242246

243247
if "%(tmp)s" in ret:
244248
tmp2 = "Tmp2_%s" % name
@@ -264,17 +268,17 @@ def as_call(
264268
call = VariableCall(
265269
call=wrapper % tmp,
266270
precall=self.convert_to_c(pkg=pkg) % {
267-
"var":name, "tmp":tmp},
271+
"var": name, "tmp": tmp},
268272
postcall=self.cleanup % tmp,
269-
tmpvars=[Local_Var(name=tmp, type=self.cparam)]
270-
+ []) # additional_tmp_vars
273+
tmpvars=[Local_Var(name=tmp, type=self.cparam)] + [])
271274

272275
elif self.cleanup:
273276
tmp = "Tmp_%s" % name
274-
conv = self.convert_to_c(pkg=pkg) % {"var":name}
277+
conv = self.convert_to_c(pkg=pkg) % {"var": name}
275278

276-
# Initialize the temporary variable with a default value, in case
277-
# it is an unconstrained type (a chars_ptr_array for instance)
279+
# Initialize the temporary variable with a default value, in
280+
# case it is an unconstrained type (a chars_ptr_array for
281+
# instance)
278282
call = VariableCall(
279283
call=wrapper % tmp,
280284
precall='',
@@ -283,7 +287,7 @@ def as_call(
283287
name=tmp, type=AdaType(self.cparam), default=conv)])
284288

285289
else:
286-
conv = self.convert_to_c(pkg=pkg) % {"var":name}
290+
conv = self.convert_to_c(pkg=pkg) % {"var": name}
287291
call = VariableCall(
288292
call=wrapper % conv, precall='', postcall="", tmpvars=[])
289293

@@ -411,7 +415,7 @@ def __init__(self, ada, userecord=True, allow_none=False, classwide=False):
411415
self.is_ptr = False
412416
self.classwide = classwide # Parameter should include "'Class"
413417
self.userecord = userecord # Parameter should be "access .._Record"
414-
self.allow_none=allow_none
418+
self.allow_none = allow_none
415419

416420
def convert_from_c(self):
417421
stub = "Stub_%s" % (base_name(self.ada), )
@@ -451,6 +455,7 @@ def copy(self):
451455
result = CType.copy(self)
452456
return result
453457

458+
454459
class Tagged(GObject):
455460
"""Tagged types that map C objects, but do not derive from GObject"""
456461

@@ -467,6 +472,7 @@ def as_ada_param(self, pkg):
467472
# Make sure to bind as a CType here, not as a GOBject
468473
return CType.as_ada_param(self, pkg)
469474

475+
470476
class UTF8(CType):
471477
def __init__(self):
472478
CType.__init__(self, "UTF8_String", "Glib.Properties.Property_String")
@@ -539,7 +545,7 @@ def add_with(self, pkg=None, specs=False):
539545
if pkg:
540546
pkg.add_with("GNAT.Strings", specs=True)
541547
pkg.add_with("Interfaces.C.Strings", specs=specs)
542-
pkg.add_with("Gtkada.Bindings", specs=specs)
548+
pkg.add_with("Gtkada.Bindings", specs=specs, might_be_unused=True)
543549

544550

545551
class Record(CType):
@@ -2048,20 +2054,26 @@ def _output_withs(self, withs):
20482054
if withs:
20492055
result = []
20502056
m = max_length(withs)
2051-
for w in sorted(withs.keys()):
2057+
had_warnings_off = False
2058+
2059+
# sort so that all packages for which 'might_be_unused' is True
2060+
# are last in the list
2061+
2062+
for w in sorted(withs.keys(), key=lambda w: 'zz%s' % w if withs[w][1] else w):
20522063
do_use, might_be_unused = withs[w]
20532064

2054-
if might_be_unused:
2065+
if might_be_unused and not had_warnings_off:
20552066
result.append("pragma Warnings(Off); -- might be unused")
2067+
had_warnings_off = True
20562068

20572069
if do_use:
20582070
result.append(
20592071
"with %-*s use %s;" % (m + 1, w + ";", w))
20602072
else:
20612073
result.append("with %s;" % w)
20622074

2063-
if might_be_unused:
2064-
result.append("pragma Warnings(On);")
2075+
if had_warnings_off:
2076+
result.append("pragma Warnings(On);")
20652077

20662078
return "\n".join(result) + "\n"
20672079
return ""

contrib/binding.py

+9-4
Original file line numberDiff line numberDiff line change
@@ -1439,7 +1439,8 @@ def _virtual_methods(self):
14391439
info = ''
14401440

14411441
for c in all:
1442-
if not self.gtkpkg.bind_virtual_method(c.get('name')):
1442+
if not self.gtkpkg.bind_virtual_method(
1443+
c.get('name'), default=self.is_interface):
14431444
continue
14441445

14451446
gtkmethod=self.gtkpkg.get_method(cname=c.get(cidentifier))
@@ -1451,7 +1452,12 @@ def _virtual_methods(self):
14511452
section = self.pkg.section("Virtual Methods")
14521453

14531454
info += 'procedure Set_%s\n' % basename
1454-
info += ' (Self : %s_Interface_Descr;\n' % ifacename
1455+
1456+
if self.is_interface:
1457+
info += ' (Self : %s_Interface_Descr;\n' % ifacename
1458+
else:
1459+
info += ' (Self : Glib.Object.GObject_Class;\n'
1460+
14551461
info += ' Handler : %s);\n' % adaname
14561462
info += 'pragma Import (C, Set_%s, "gtkada_%s_set_%s");\n' % (
14571463
basename, ifacename, basename.lower())
@@ -2698,8 +2704,7 @@ def generate(self, gir):
26982704
self._globals()
26992705
self._fields()
27002706

2701-
if self.is_interface:
2702-
self._virtual_methods()
2707+
self._virtual_methods()
27032708

27042709
if not into and self.name != "Gtk.Widget":
27052710
self._implements()

contrib/binding.xml

+2
Original file line numberDiff line numberDiff line change
@@ -7468,6 +7468,8 @@ package body Generic_Set is
74687468
</package>
74697469

74707470
<package id="GApplication" ada="Glib.Application">
7471+
<virtual-method id='command_line' bind='True'/>
7472+
<virtual-method id='local_command_line' bind='True'/>
74717473
<enum ctype="GApplicationFlags"/>
74727474
<method id="g_application_get_dbus_connection" bind="False"/> <!-- requires DBus -->
74737475
<method id="g_application_open" bind="False" /> <!-- requires GFile -->

contrib/binding_gtkada.py

+16-11
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,10 @@
4343
direction="..." <!-- Override direction (see <parameter> node below) -->
4444
/>
4545
46+
<!-- By default, virtual methods are bound for interfaces, but not for
47+
other classes. You should use the following to control which
48+
methods should be bound.
49+
-->
4650
<virtual-method <!-- list of virtual methods to bind -->
4751
id='...' <!-- the name of the virtual method, or '*' -->
4852
bind=True <!-- by default, all methods bound for interfaces -->
@@ -437,20 +441,21 @@ def get_global_functions(self):
437441
if c.get("bind", "true").lower() != "false"]
438442
return []
439443

440-
def bind_virtual_method(self, name):
444+
def bind_virtual_method(self, name, default):
441445
"""
442446
Whether to bind the given virtual method
443447
"""
444-
if not hasattr(self, 'disabled_virtual_methods'):
445-
if self.node is None:
446-
self.disabled_virtual_methods = set()
447-
else:
448-
self.disabled_virtual_methods = set(
449-
c.get('id')
450-
for c in self.node.findall("virtual-method")
451-
if c.get("bind", "true").lower() == "false")
452-
return (name not in self.disabled_virtual_methods
453-
and '*' not in self.disabled_virtual_methods)
448+
if not hasattr(self, 'virtual_methods'):
449+
self.virtual_methods = {'*': default}
450+
if self.node is not None:
451+
for c in self.node.findall("virtual-method"):
452+
self.virtual_methods[c.get('id')] = (
453+
c.get("bind", "true").lower() == "true")
454+
455+
v = self.virtual_methods.get(name, None)
456+
if v is None:
457+
v = self.virtual_methods['*']
458+
return v
454459

455460

456461
class GtkAdaMethod(object):

src/generated/glib-action_group.ads

+1-1
Original file line numberDiff line numberDiff line change
@@ -76,8 +76,8 @@ with Glib; use Glib;
7676
with Glib.Object; use Glib.Object;
7777
with Glib.Types; use Glib.Types;
7878
with Glib.Variant; use Glib.Variant;
79-
with Gtkada.Bindings; use Gtkada.Bindings;
8079
pragma Warnings(Off); -- might be unused
80+
with Gtkada.Bindings; use Gtkada.Bindings;
8181
with Interfaces.C.Strings; use Interfaces.C.Strings;
8282
pragma Warnings(On);
8383

src/generated/glib-application.adb

-4
Original file line numberDiff line numberDiff line change
@@ -28,10 +28,6 @@ with Glib.Type_Conversion_Hooks; use Glib.Type_Conversion_Hooks;
2828
with Glib.Values; use Glib.Values;
2929
with Gtk.Arguments; use Gtk.Arguments;
3030
with GtkAda.Types; use GtkAda.Types;
31-
with Gtkada.Bindings; use Gtkada.Bindings;
32-
pragma Warnings(Off); -- might be unused
33-
with Interfaces.C.Strings; use Interfaces.C.Strings;
34-
pragma Warnings(On);
3531

3632
package body Glib.Application is
3733

src/generated/glib-application.ads

+40
Original file line numberDiff line numberDiff line change
@@ -167,6 +167,10 @@ with Glib.Object; use Glib.Object;
167167
with Glib.Properties; use Glib.Properties;
168168
with Glib.Types; use Glib.Types;
169169
with Glib.Variant; use Glib.Variant;
170+
with Interfaces.C.Strings; use Interfaces.C.Strings;
171+
pragma Warnings(Off); -- might be unused
172+
with Gtkada.Bindings; use Gtkada.Bindings;
173+
pragma Warnings(On);
170174

171175
package Glib.Application is
172176

@@ -862,6 +866,42 @@ package Glib.Application is
862866
return Gapplication
863867
renames Implements_Gaction_Map.To_Object;
864868

869+
---------------------
870+
-- Virtual Methods --
871+
---------------------
872+
873+
type Virtual_Command_Line is access function
874+
(Self : System.Address;
875+
Command_Line : System.Address) return Gint;
876+
pragma Convention (C, Virtual_Command_Line);
877+
878+
type Virtual_Local_Command_Line is access function
879+
(Self : System.Address;
880+
Arguments : access Interfaces.C.Strings.chars_ptr_array;
881+
Exit_Status : access Gint) return Glib.Gboolean;
882+
pragma Convention (C, Virtual_Local_Command_Line);
883+
-- This virtual function is always invoked in the local instance. It gets
884+
-- passed a pointer to a null-terminated copy of Argv and is expected to
885+
-- remove arguments that it handled (shifting up remaining arguments).
886+
-- The last argument to local_command_line is a pointer to the Status
887+
-- variable which can used to set the exit status that is returned from
888+
-- Glib.Application.Run.
889+
-- See Glib.Application.Run for more details on
890+
-- Glib.Application.Gapplication startup.
891+
-- "arguments": array of command line arguments
892+
-- "exit_status": exit status to fill after processing the command line.
893+
894+
subtype Application_Interface_Descr is Glib.Object.Interface_Description;
895+
procedure Set_Command_Line
896+
(Self : Glib.Object.GObject_Class;
897+
Handler : Virtual_Command_Line);
898+
pragma Import (C, Set_Command_Line, "gtkada_Application_set_command_line");
899+
procedure Set_Local_Command_Line
900+
(Self : Glib.Object.GObject_Class;
901+
Handler : Virtual_Local_Command_Line);
902+
pragma Import (C, Set_Local_Command_Line, "gtkada_Application_set_local_command_line");
903+
-- See Glib.Object.Add_Interface
904+
865905
private
866906
Platform_Data_Property : constant Glib.Properties.Property_Object :=
867907
Glib.Properties.Build ("platform-data");

src/generated/glib-simple_action_group.adb

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,8 +24,8 @@
2424
pragma Style_Checks (Off);
2525
pragma Warnings (Off, "*is already use-visible*");
2626
with Glib.Type_Conversion_Hooks; use Glib.Type_Conversion_Hooks;
27-
with Gtkada.Bindings; use Gtkada.Bindings;
2827
pragma Warnings(Off); -- might be unused
28+
with Gtkada.Bindings; use Gtkada.Bindings;
2929
with Interfaces.C.Strings; use Interfaces.C.Strings;
3030
pragma Warnings(On);
3131

src/generated/gtk-application_window.adb

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,8 +24,8 @@
2424
pragma Style_Checks (Off);
2525
pragma Warnings (Off, "*is already use-visible*");
2626
with Glib.Type_Conversion_Hooks; use Glib.Type_Conversion_Hooks;
27-
with Gtkada.Bindings; use Gtkada.Bindings;
2827
pragma Warnings(Off); -- might be unused
28+
with Gtkada.Bindings; use Gtkada.Bindings;
2929
with Interfaces.C.Strings; use Interfaces.C.Strings;
3030
pragma Warnings(On);
3131

src/generated/gtk-font_chooser.ads

+3-3
Original file line numberDiff line numberDiff line change
@@ -37,12 +37,12 @@ with Glib; use Glib;
3737
with Glib.Object; use Glib.Object;
3838
with Glib.Properties; use Glib.Properties;
3939
with Glib.Types; use Glib.Types;
40-
pragma Warnings(Off); -- might be unused
41-
with Interfaces.C.Strings; use Interfaces.C.Strings;
42-
pragma Warnings(On);
4340
with Pango.Font; use Pango.Font;
4441
with Pango.Font_Face; use Pango.Font_Face;
4542
with Pango.Font_Family; use Pango.Font_Family;
43+
pragma Warnings(Off); -- might be unused
44+
with Interfaces.C.Strings; use Interfaces.C.Strings;
45+
pragma Warnings(On);
4646

4747
package Gtk.Font_Chooser is
4848

src/generated/gtk-selection_data.adb

+3-1
Original file line numberDiff line numberDiff line change
@@ -24,9 +24,11 @@
2424
pragma Style_Checks (Off);
2525
pragma Warnings (Off, "*is already use-visible*");
2626
with GtkAda.Types; use GtkAda.Types;
27-
with Gtkada.Bindings; use Gtkada.Bindings;
2827
with Gtkada.C; use Gtkada.C;
2928
with Interfaces.C.Strings; use Interfaces.C.Strings;
29+
pragma Warnings(Off); -- might be unused
30+
with Gtkada.Bindings; use Gtkada.Bindings;
31+
pragma Warnings(On);
3032

3133
package body Gtk.Selection_Data is
3234

src/misc_generated.c

+6
Original file line numberDiff line numberDiff line change
@@ -313,4 +313,10 @@ void gtkada_Action_Map_set_lookup_action(GActionMapInterface* iface, void* handl
313313
}
314314
void gtkada_Action_Map_set_remove_action(GActionMapInterface* iface, void* handler) {
315315
iface->remove_action = handler;
316+
}
317+
void gtkada_Application_set_command_line(GApplicationClass* iface, void* handler) {
318+
iface->command_line = handler;
319+
}
320+
void gtkada_Application_set_local_command_line(GApplicationClass* iface, void* handler) {
321+
iface->local_command_line = handler;
316322
}

0 commit comments

Comments
 (0)