-dnl Copyright © 2019-2020 Nick Bowler
+dnl Copyright © 2019-2023 Nick Bowler
dnl License GPLv2+: GNU General Public License version 2 or any later version.
dnl This is free software: you are free to change and redistribute it.
dnl There is NO WARRANTY, to the extent permitted by law.
AT_SETUP([str.gob])
+# check if compiler supports format warnings
+AT_DATA([fmt.c],
+[[#include <config.h>
+#include <glib.h>
+
+void foo(const char *, ...) G_GNUC_PRINTF(1, 2);
+void bar(void) {
+#line 99 "VERIFY"
+foo("%s", 0);
+}
+]])
+AT_DATA([fmt.awk],
+[[BEGIN { pass=0; FS=":"; }
+/%s/ && $1 == "VERIFY" && $2 == 99 { pass=1; }
+END { exit(!pass); }
+]])
+
+fmt_warnings=false
+set x $CPPFLAGS $CFLAGS $LIBGOBJECT_CFLAGS; shift
+AS_IF([$CC "$@" -c fmt.c 1>/dev/null 2>out && $AWK -f fmt.awk out],
+ [fmt_warnings=:],
+ [AS_IF([$CC "$@" -Wformat -c fmt.c 1>/dev/null 2>out && $AWK -f fmt.awk out],
+ [CFLAGS="$CFLAGS -Wformat" fmt_warnings=:])])
+
AT_DATA([main.c],
-[[#include "str.h"
+[[#include <config.h>
+#include "str.h"
int main(void)
{
- g_type_init ();
-
- int the_answer = 42;
+ Str *test_good, *test_bad;
char *stupid_pointer = "ug";
+ int the_answer = 42;
+
+ g_type_init ();
/* This works fine. */
- Str *test_good = (Str *) (str_new ("%d", the_answer));
+ test_good = (Str *) (str_new ("%d", the_answer));
test_good = test_good;
/* This gets a warning thanks to our function attribute. */
- Str *test_bad = (Str *) (str_new ("%d", stupid_pointer));
+ test_bad = (Str *) (str_new ("%d", stupid_pointer));
test_bad = test_bad;
return 0;
str_gob=$abs_top_srcdir/t/str.gob
AT_CHECK([gob2 "$str_gob"])
+AT_CHECK([$HAVE_GOBJECT_PRIVATES || exit 77])
+
TEST_COMPILE_GOBJECT([str.c], [0], [], [stderr])
mv stderr str_stderr
+TEST_COMPILE_GOBJECT([main.c], [0], [], [stderr])
+mv stderr main_stderr
+TEST_LINK_GOBJECT([main], [str.o main.o])
+
+AT_DATA([str.awk],
+[[/want a string/ { lines[NR] = 1; }
+END {
+ FS=":";
+ while ((rc = getline < "str_stderr") > 0) {
+ sub(/.*[\/]/, "", $1);
+ if (/%s/ && $1 == "str.gob" && $2 in lines)
+ lines[$2]--;
+ }
-# Check for correct diagnostic messages on the target lines...
-AT_CHECK([awk '/want a string/ { print NR }' "$str_gob" >str_lines])
-total=0
-exec 3<str_lines
-while read l <&3; do
- AS_VAR_ARITH([total], [1 + $total])
- AT_CHECK([awk -v line="$l" -v file="$str_gob" -NF : \
- '$1 == file && $2 == line { exit 42 }' str_stderr], [42])
-done
-exec 3<&-
-AT_CHECK([test x"$total" = x"2"])
+ if (rc < 0)
+ exit(1);
-TEST_COMPILE_GOBJECT([main.c], [0], [], [stderr])
-AT_CHECK([awk -NF : '$1 == "main.c" && $2 == "15" { exit 42 }' stderr], [42])
+ count=0;
+ for (l in lines) {
+ if (lines[l])
+ exit(1);
+ count++;
+ }
-AT_CHECK([$CC $CFLAGS $LDFLAGS $LIBGOBJECT_LIBS -o main str.o main.o])
+ exit(count != 2);
+}
+]])
+
+# Check for correct diagnostic messages on the target lines...
+AT_CHECK([$fmt_warnings || exit 77
+$AWK -f str.awk "$str_gob" || exit 1
+$AWK -f - main_stderr <<'EOF'
+BEGIN { pass=0; FS=":"; }
+/%d/ && $1 == "main.c" && $2 == 17 { pass=1; }
+END { exit(!pass); }
+EOF])
AT_CLEANUP
AT_SETUP([dynamic types])
AT_KEYWORDS([dynamic])
-AT_DATA([test.gob],
-[[class :Test from G:Object (dynamic)
+AT_DATA([test.gob], [[%ctop{
+#include <config.h>
+%}
+class :Test from G:Object (dynamic)
{
public void test(void)
{
}
]])
AT_CHECK([gob2 test.gob])
+TEST_COMPILE_GOBJECT([test.c], [0], [], [ignore])
AT_DATA([main.c],
-[[#include "test.h"
+[[#include <config.h>
+#include "test.h"
int main(void)
{
+ g_type_init();
test_register_type(NULL);
}
]])
-
-TEST_COMPILE_GOBJECT([test.c], [0], [], [ignore])
TEST_COMPILE_GOBJECT([main.c], [0], [], [ignore])
-AT_CHECK([$CC $CFLAGS $LDFLAGS $LIBGOBJECT_LIBS -o main test.o main.o])
+TEST_LINK_GOBJECT([main], [test.o main.o])
AT_CLEANUP
AT_SETUP([dynamic type registration])
AT_KEYWORDS([dynamic runtime])
-AT_DATA([test.gob],
-[[%{
+AT_DATA([test.gob], [[%ctop{
+#include <config.h>
+%}
+%{
#include <stdio.h>
%}
class :Test from G:Object (dynamic)
}
}
]])
+AT_CHECK([gob2 test.gob])
+TEST_COMPILE_GOBJECT([test.c], [0], [], [ignore])
-AT_DATA([mod.gob],
-[[%{
-#include "test.h"
-%}
-class :Mod from G:Type:Module
-{
- override (G:Type:Module) gboolean load(G:Type:Module *m)
- {
- test_register_type(m);
- return TRUE;
- }
-}
-]])
+TEST_TYPE_MODULE([:Test])
AT_DATA([main.c],
-[[#include <stdio.h>
-#include "mod.h"
+[[#include <stdlib.h>
#include "test.h"
+#include "test-mod.h"
void devnull(const char *a, GLogLevelFlags b, const char *c, gpointer d) { }
int main(void)
{
- GTypeModule *m = g_object_new(mod_get_type(), NULL);
guint handler;
Test *t;
+ g_type_init();
+
/* should fail, suppress internal glib logging... */
handler = g_log_set_handler("GLib-GObject", G_LOG_LEVEL_MASK, devnull, 0);
t = g_object_new(test_get_type(), NULL);
return EXIT_FAILURE;
g_log_remove_handler("GLib-GObject", handler);
- g_type_module_use(m);
+ /* Register dynamic type */
+ g_type_module_use(g_object_new(test_mod_get_type(), NULL));
+
/* should work now */
t = g_object_new(test_get_type(), "s", "Hello, World", (char *)NULL);
if (!t)
return EXIT_SUCCESS;
}
]])
-
-AT_CHECK([gob2 mod.gob])
-AT_CHECK([gob2 test.gob])
-TEST_COMPILE_GOBJECT([mod.c], [0], [], [ignore])
-TEST_COMPILE_GOBJECT([test.c], [0], [], [ignore])
TEST_COMPILE_GOBJECT([main.c], [0], [], [ignore])
-AT_CHECK([$CC $CFLAGS $LDFLAGS $LIBGOBJECT_LIBS -o main mod.o test.o main.o])
+TEST_LINK_GOBJECT([main], [test.o test-mod.o main.o])
AT_CHECK([./main], [0], [Hello, World
])
AT_CLEANUP
-dnl Dynamic types: check that we can call interface methods of dynamic types.
-AT_SETUP([dynamic interface implementation])
-AT_KEYWORDS([dynamic runtime interface])
-
-AT_DATA([iface.h],
-[[#define IF_TYPE_TEST if_test_get_type()
-
-typedef struct IFTestIface {
- GTypeInterface parent;
-
- void (*test)(GObject *obj);
-} IFTestIface;
-
-GType if_test_get_type(void);
-]])
-
-AT_DATA([test.gob],
-[[%{
-#include <stdio.h>
-#include "iface.h"
-%}
-
-class :Test from G:Object (dynamic)
- (interface IF:Test)
-{
- private const char *s = { "Hello, World!" };
- interface IF:Test private void test(G:Object *o)
- {
- Self *self = SELF(o);
- printf("%s\n", self->_priv->s);
- }
-}
-]])
-
-AT_DATA([mod.gob],
-[[%{
-#include "test.h"
-%}
-class :Mod from G:Type:Module
-{
- override (G:Type:Module) gboolean load(G:Type:Module *m)
- {
- test_register_type(m);
- return TRUE;
- }
-}
-]])
-
-AT_DATA([main.c],
-[[#include "test.h"
-#include "mod.h"
-#include "iface.h"
-
-GType if_test_get_type(void)
-{
- static GType type = 0;
- if (type == 0) {
- static const GTypeInfo info = {
- sizeof (IFTestIface),
- NULL,
- NULL,
- };
-
- type = g_type_register_static(G_TYPE_INTERFACE, "IFTest", &info, 0);
- }
- return type;
-}
-
-int main(void)
-{
- GTypeModule *m = g_object_new(TYPE_MOD, NULL);
- GObject *t;
-
- g_type_module_use(m);
- t = g_object_new(TYPE_TEST, NULL);
-
- g_return_val_if_fail(G_TYPE_CHECK_INSTANCE_TYPE(t, IF_TYPE_TEST),
- EXIT_FAILURE);
- G_TYPE_INSTANCE_GET_INTERFACE(t, IF_TYPE_TEST, IFTestIface)->test(t);
-
- return EXIT_SUCCESS;
-}
-]])
-
-AT_CHECK([gob2 mod.gob])
-AT_CHECK([gob2 test.gob])
-TEST_COMPILE_GOBJECT([mod.c], [0], [], [ignore])
-TEST_COMPILE_GOBJECT([test.c], [0], [], [ignore])
-TEST_COMPILE_GOBJECT([main.c], [0], [], [ignore])
-AT_CHECK([$CC $CFLAGS $LDFLAGS $LIBGOBJECT_LIBS -o main mod.o test.o main.o])
-AT_CHECK([./main], [0], [Hello, World!
-])
-
-AT_CLEANUP
-
AT_SETUP([GOB2_CHECK min-version test])
AT_DATA([configure.ac],
cat >configure.ac <<EOF
[m4@&t@_include([$builddir/gob2.m4])]
[m4@&t@_pattern_forbid([^GOB2_])]
+[m4@&t@_pattern_forbid([^DX_])]
[AC_INIT([gob2_check], [0])]
[GOB2_CHECK(]m4_dquote(m4_defn([MYVER]))[)]
GOB2=\`command -v \$GOB2\`
AT_CHECK([./configure], [1], [ignore], [ignore])
AT_CLEANUP
+
+AT_SETUP([private data members])
+
+AT_DATA([test.gob], [[%ctop{
+#include <config.h>
+%}
+class :Test from G:Object
+{
+ private int x = 42;
+
+ public int get_x(G:Object *go) { return TEST(go)->_priv->x; }
+ public void set_x(G:Object *go, int val) { TEST(go)->_priv->x = val; }
+}
+]])
+AT_CHECK([gob2 test.gob])
+AT_CHECK([$HAVE_GOBJECT_PRIVATES || exit 77])
+TEST_COMPILE_GOBJECT([test.c], [0], [], [ignore])
+
+AT_DATA([main.c], [[#include <config.h>
+#include <stdio.h>
+#include "test.h"
+
+int main(void)
+{
+ GObject *go;
+
+ g_type_init();
+ go = g_object_new(test_get_type(), NULL);
+
+ printf("%d\n", test_get_x(go));
+ printf("%d\n", (test_set_x(go, 123), test_get_x(go)));
+
+ return 0;
+}
+]])
+TEST_COMPILE_GOBJECT([main.c], [0], [], [ignore])
+TEST_LINK_GOBJECT([main], [test.o main.o])
+AT_CHECK([./main], [0], [42
+123
+])
+
+AT_CLEANUP
+
+AT_SETUP([private data members (dynamic)])
+
+AT_DATA([test.gob], [[%ctop{
+#include <config.h>
+%}
+class :Test from G:Object (dynamic)
+{
+ private int x = 54;
+
+ public int get_x(G:Object *go) { return TEST(go)->_priv->x; }
+ public void set_x(G:Object *go, int val) { TEST(go)->_priv->x = val; }
+}
+]])
+AT_CHECK([gob2 test.gob])
+AT_CHECK([$HAVE_GOBJECT_PRIVATES || exit 77])
+TEST_COMPILE_GOBJECT([test.c], [0], [], [ignore])
+
+TEST_TYPE_MODULE([:Test])
+
+AT_DATA([main.c], [[#include <config.h>
+#include <stdio.h>
+#include "test.h"
+#include "test-mod.h"
+
+int main(void)
+{
+ GObject *go;
+
+ g_type_init();
+ g_type_module_use(g_object_new(test_mod_get_type(), NULL));
+ go = g_object_new(test_get_type(), NULL);
+
+ printf("%d\n", test_get_x(go));
+ printf("%d\n", (test_set_x(go, 123), test_get_x(go)));
+
+ return 0;
+}
+]])
+TEST_COMPILE_GOBJECT([main.c], [0], [], [ignore])
+TEST_LINK_GOBJECT([main], [test.o test-mod.o main.o])
+AT_CHECK([./main], [0], [54
+123
+])
+
+AT_CLEANUP