Reportedly in the new PPC ELF v2 ABI calls to varargs functions act
differently from calls to normal functions.  The Go compiler can call C
functions directly using //extern comments, but it can't represent a C
varargs function.  Rather than extend the compiler, this patch changes
the library so that C varargs functions are only called from C code, by
introducing a non-varargs C function intermediary that Go can call and
that simply calls the C function.  Bootstrapped and ran Go testsuite on
x86_64-unknown-linux-gnu.  Committed to mainline.

Ian

diff -r dc9e1cbc4013 libgo/Makefile.am
--- a/libgo/Makefile.am	Sat Nov 23 10:55:58 2013 -0800
+++ b/libgo/Makefile.am	Sat Nov 23 18:34:48 2013 -0800
@@ -488,6 +488,7 @@
 	runtime/go-unsafe-newarray.c \
 	runtime/go-unsafe-pointer.c \
 	runtime/go-unwind.c \
+	runtime/go-varargs.c \
 	runtime/chan.c \
 	runtime/cpuprof.c \
 	runtime/env_posix.c \
diff -r dc9e1cbc4013 libgo/configure.ac
--- a/libgo/configure.ac	Sat Nov 23 10:55:58 2013 -0800
+++ b/libgo/configure.ac	Sat Nov 23 18:34:48 2013 -0800
@@ -506,7 +506,7 @@
 AM_CONDITIONAL(HAVE_STRERROR_R, test "$ac_cv_func_strerror_r" = yes)
 AM_CONDITIONAL(HAVE_WAIT4, test "$ac_cv_func_wait4" = yes)
 
-AC_CHECK_FUNCS(accept4 dup3 epoll_create1 faccessat fallocate fchmodat fchownat futimesat getxattr inotify_add_watch inotify_init inotify_init1 inotify_rm_watch listxattr mkdirat mknodat openat pipe2 removexattr renameat setxattr sync_file_range splice tee unlinkat unshare utimensat)
+AC_CHECK_FUNCS(accept4 dup3 epoll_create1 faccessat fallocate fchmodat fchownat futimesat getxattr inotify_add_watch inotify_init inotify_init1 inotify_rm_watch listxattr mkdirat mknodat open64 openat pipe2 removexattr renameat setxattr sync_file_range splice tee unlinkat unshare utimensat)
 AC_TYPE_OFF_T
 AC_CHECK_TYPES([loff_t])
 
diff -r dc9e1cbc4013 libgo/go/syscall/exec_unix.go
--- a/libgo/go/syscall/exec_unix.go	Sat Nov 23 10:55:58 2013 -0800
+++ b/libgo/go/syscall/exec_unix.go	Sat Nov 23 18:34:48 2013 -0800
@@ -30,7 +30,7 @@
 //chdir(path *byte) _C_int
 
 //sysnb	raw_fcntl(fd int, cmd int, arg int) (val int, err Errno)
-//fcntl(fd _C_int, cmd _C_int, arg _C_int) _C_int
+//__go_fcntl(fd _C_int, cmd _C_int, arg _C_int) _C_int
 
 //sysnb	raw_close(fd int) (err Errno)
 //close(fd _C_int) _C_int
diff -r dc9e1cbc4013 libgo/go/syscall/libcall_linux.go
--- a/libgo/go/syscall/libcall_linux.go	Sat Nov 23 10:55:58 2013 -0800
+++ b/libgo/go/syscall/libcall_linux.go	Sat Nov 23 18:34:48 2013 -0800
@@ -9,7 +9,7 @@
 import "unsafe"
 
 //sys	Openat(dirfd int, path string, flags int, mode uint32) (fd int, err error)
-//openat(dirfd _C_int, path *byte, flags _C_int, mode Mode_t) _C_int
+//__go_openat(dirfd _C_int, path *byte, flags _C_int, mode Mode_t) _C_int
 
 //sys	futimesat(dirfd int, path *byte, times *[2]Timeval) (err error)
 //futimesat(dirfd _C_int, path *byte, times *[2]Timeval) _C_int
diff -r dc9e1cbc4013 libgo/go/syscall/libcall_posix.go
--- a/libgo/go/syscall/libcall_posix.go	Sat Nov 23 10:55:58 2013 -0800
+++ b/libgo/go/syscall/libcall_posix.go	Sat Nov 23 18:34:48 2013 -0800
@@ -206,7 +206,7 @@
 //fchown(fd _C_int, uid Uid_t, gid Gid_t) _C_int
 
 //sys	fcntl(fd int, cmd int, arg int) (val int, err error)
-//fcntl(fd _C_int, cmd _C_int, arg _C_int) _C_int
+//__go_fcntl(fd _C_int, cmd _C_int, arg _C_int) _C_int
 
 //sys	Fdatasync(fd int) (err error)
 //fdatasync(fd _C_int) _C_int
diff -r dc9e1cbc4013 libgo/go/syscall/libcall_posix_largefile.go
--- a/libgo/go/syscall/libcall_posix_largefile.go	Sat Nov 23 10:55:58 2013 -0800
+++ b/libgo/go/syscall/libcall_posix_largefile.go	Sat Nov 23 18:34:48 2013 -0800
@@ -19,7 +19,7 @@
 //mmap64(addr *byte, length Size_t, prot _C_int, flags _C_int, fd _C_int, offset Offset_t) *byte
 
 //sys	Open(path string, mode int, perm uint32) (fd int, err error)
-//open64(path *byte, mode _C_int, perm Mode_t) _C_int
+//__go_open64(path *byte, mode _C_int, perm Mode_t) _C_int
 
 //sys	Pread(fd int, p []byte, offset int64) (n int, err error)
 //pread64(fd _C_int, buf *byte, count Size_t, offset Offset_t) Ssize_t
diff -r dc9e1cbc4013 libgo/go/syscall/libcall_posix_regfile.go
--- a/libgo/go/syscall/libcall_posix_regfile.go	Sat Nov 23 10:55:58 2013 -0800
+++ b/libgo/go/syscall/libcall_posix_regfile.go	Sat Nov 23 18:34:48 2013 -0800
@@ -20,7 +20,7 @@
 //mmap(addr *byte, length Size_t, prot _C_int, flags _C_int, fd _C_int, offset Offset_t) *byte
 
 //sys	Open(path string, mode int, perm uint32) (fd int, err error)
-//open(path *byte, mode _C_int, perm Mode_t) _C_int
+//__go_open(path *byte, mode _C_int, perm Mode_t) _C_int
 
 //sys	Pread(fd int, p []byte, offset int64) (n int, err error)
 //pread(fd _C_int, buf *byte, count Size_t, offset Offset_t) Ssize_t
diff -r dc9e1cbc4013 libgo/runtime/go-varargs.c
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libgo/runtime/go-varargs.c	Sat Nov 23 18:34:48 2013 -0800
@@ -0,0 +1,47 @@
+/* go-varargs.c -- functions for calling C varargs functions.
+
+   Copyright 2013 The Go Authors. All rights reserved.
+   Use of this source code is governed by a BSD-style
+   license that can be found in the LICENSE file.  */
+
+#include "config.h"
+
+#include <sys/types.h>
+#include <fcntl.h>
+
+/* The syscall package calls C functions.  The Go compiler can not
+   represent a C varargs functions.  On some systems it's important
+   that the declaration of a function match the call.  This function
+   holds non-varargs C functions that the Go code can call.  */
+
+int
+__go_open (char *path, int mode, mode_t perm)
+{
+  return open (path, mode, perm);
+}
+
+int
+__go_fcntl (int fd, int cmd, int arg)
+{
+  return fcntl (fd, cmd, arg);
+}
+
+#ifdef HAVE_OPEN64
+
+int
+__go_open64 (char *path, int mode, mode_t perm)
+{
+  return open64 (path, mode, perm);
+}
+
+#endif
+
+#ifdef HAVE_OPENAT
+
+int
+__go_openat (int fd, char *path, int flags, mode_t mode)
+{
+  return openat (fd, path, flags, mode);
+}
+
+#endif

Reply via email to