We know that C code can be called from Go through Cgo, so can we call Go from C? The answer is yes, through dynamically linked libraries.

Before we get into the details, let’s discuss a usage scenario for calling Go through C. Compared with C, Go has the advantage of high development efficiency, especially for writing networks. If some old systems have been written in C/C++, patch needs to be made, and this has no special requirements on performance, but has requirements on time, then it is also a good method to use Go for development and call it through C.

0. Dynamic links

Let’s look at dynamic linking first, and static linking. We know that a C program (such as main.c) on the GNU compilation system has to go through several steps from code to execution: pre-processing, compilation, assembly, and linking. You can see these steps when compiling with GCC with the -v argument (there may be no preprocessing step, since many compilers now incorporate preprocessing into the compile step). The functions of each step are as follows:

  1. Preprocessing: Code replacement of references in code. The main. C – > main i..
  2. Compile: Translate main. I into an assembler file main.s.
  3. Assembler: Translate main.s into a relocatable object file main.o.
  4. Link: Combine main.o and other dependent modules into an executable object file.

Static linking takes the set of relocatable object files generated in step 3 above and produces a fully linked executable file that can be loaded and run as output. There are two main steps in this process: symbol resolution and relocation. Symbols include functions, variables, etc. Symbol parsing is to associate symbols in other modules referenced in our code with their definitions. Relocation is to ensure that the program can be executed to the correct memory location when running. In layman’s terms: static linking is a combination of multiple module object files into an executable object file.

The downside of static linking is that every module our program depends on needs to be integrated into the final executable file, and if some of the base modules are shared by everyone’s program, it means multiple copies in memory, which is a waste of time. In addition, every dependency module change requires our executable to compile the link again, which is quite unreasonable. For example, if the operating system is completely statically linked, then every operating system patch means that we have to recompile the operating system code, but it doesn’t, because the patch is dynamically linked.

The dynamic link input is no longer the object file.o, but the dynamic link library.so. The compiler sees.so and says, “Oh, this function address will be determined at runtime.” So how do you know for sure? In the case of the GNU compilation system, the technique used is lazy binding. Deferred binding is implemented through two key data structures: the Global Offset Table (GOT) and the Procedure Linkage Table (PLT). Interested students can do their own search, or refer to chapter 7 “Links” in Understanding Computer Systems.

With that said, let’s start by building our own so file in C. The first is the header file: legendtkl.h.

//legendtkl.h
int foo();
int bar();
Copy the code

Here is the.c file: legendtkl.c.

#include "legendtkl.h" #include <stdio.h> int foo() {int a[1000] = {1,2,3}; return a[0] + a[1] + a[2]; } int bar() { printf("hello, I am Legendtkl"); return 42; }Copy the code

Write another test program: test.c.

#include "legendtkl.h"

int main() {
    foo();
    bar();
    return 0;
}
Copy the code

Statically linked (-v to see the steps) :

$ gcc -v test.c legendtkl.c -o test1
Copy the code

Dynamic linking requires building our own SO file first.

$ gcc -shared -fpic -o liblegendtkl.so legendtkl.c
Copy the code

And then compile.

$ gcc -v test.c -o test2 ./liblegendtkl.so
Copy the code

Test1 is the executable generated by static links. Test2 is the object file generated by dynamic links. Test1 runs when moved anywhere; The test2 run directory must have liblegendtkl.so (so can pass absolute path). Not surprisingly: test1 has a larger file size than test2.

Another benefit of dynamic linking is that we modify legendtkl.c and compile it into so file and replace the previous so file. Running Test2 shows that we already have the new functionality, after all, dynamic linking.

1. Go code compiles to so

It’s also easy to compile Go code into SO, starting with the Go code.

package main import "C" import ( "fmt" ) //export Foo func Foo(a, b int) int { return a + b; } //export Bar func Bar() { fmt.Println("I am bar, not foo!" ) } func main() {}Copy the code

Build the so file. Go builds a larger so file because Go has runtime.

$ go build -o legendtkl.so -buildmode=c-shared legendtkl.go
Copy the code

Legendtkl. h and legendtkl.so are normally generated. The header file is responsible for doing some type conversions. Excerpts are as follows.

#ifndef GO_CGO_PROLOGUE_H
#define GO_CGO_PROLOGUE_H

typedef signed char GoInt8;
typedef unsigned char GoUint8;
typedef short GoInt16;
typedef unsigned short GoUint16;
typedef int GoInt32;
typedef unsigned int GoUint32;
typedef long long GoInt64;
typedef unsigned long long GoUint64;
typedef GoInt64 GoInt;
typedef GoUint64 GoUint;
Copy the code

C code reference

Write the test code: test.c.

#include "legendtkl.h" #include <stdio.h> int main() {printf("% LLD \n",Foo(1,2)); Bar(); return 0; }Copy the code

compile

$ gcc test.c -o test ./legendtkl.so
Copy the code

run

./test
3
I am bar, not foo!
Copy the code

3. The conclusion

After the so file is generated, other languages can also be called, so I won’t go into details here.

This column is open for contributions, and you are welcome to contribute original articles related to Go.