Merge pull request #505 from xushiwei/q
library: cpp/std (and more c++ mechanism, eg. build: clFiles support cflags)
This commit is contained in:
@@ -9,5 +9,5 @@ func main() {
|
||||
}
|
||||
|
||||
/* Expected output:
|
||||
Hello World
|
||||
Hello world
|
||||
*/
|
||||
|
||||
@@ -1,3 +1,19 @@
|
||||
/*
|
||||
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package socket
|
||||
|
||||
import (
|
||||
|
||||
@@ -1 +1,14 @@
|
||||
brew install inih
|
||||
LLGo wrapper of benhoyt/inih
|
||||
=====
|
||||
|
||||
## How to install
|
||||
|
||||
### on macOS (Homebrew)
|
||||
|
||||
```sh
|
||||
brew install inih
|
||||
```
|
||||
|
||||
### on Linux (Debian/Ubuntu)
|
||||
|
||||
TODO
|
||||
|
||||
@@ -1,61 +0,0 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"github.com/goplus/llgo/c"
|
||||
"github.com/goplus/llgo/cpp/inih"
|
||||
)
|
||||
|
||||
/*
|
||||
expected output:
|
||||
section: owner name: name value: John Doe
|
||||
section: owner name: organization value: Acme Widgets Inc.
|
||||
section: database name: server value: 192.0.2.62
|
||||
section: database name: port value: 143
|
||||
section: database name: file value: "payroll.dat"
|
||||
section: database name: enabled value: true
|
||||
section: owner name: name value: John Doe
|
||||
section: owner name: organization value: Acme Widgets Inc.
|
||||
section: database name: server value: 192.0.2.62
|
||||
section: database name: port value: 143
|
||||
section: database name: file value: "payroll.dat"
|
||||
section: database name: enabled value: true
|
||||
section: settings name: username value: lee
|
||||
section: settings name: timeout value: 20
|
||||
Config file parsed successfully
|
||||
s : 192.0.2.62
|
||||
isDatabaseEnabled: true port: 143
|
||||
0x109f54da0
|
||||
0
|
||||
value: 100
|
||||
*/
|
||||
|
||||
func main() {
|
||||
demo1()
|
||||
reader := inih.NewReaderFile(inih.Str("config.ini"))
|
||||
if reader.ParseError() != 0 {
|
||||
println("Error parsing config file")
|
||||
return
|
||||
}
|
||||
isDatabaseEnabled := reader.GetBoolean(inih.Str("database"), inih.Str("enabled"), false)
|
||||
port := reader.GetInteger(inih.Str("database"), inih.Str("port"), 0)
|
||||
s := reader.GetString(inih.Str("database"), inih.Str("server"), inih.Str("unknown"))
|
||||
println("s :", s.String())
|
||||
println("isDatabaseEnabled:", isDatabaseEnabled, "port:", port)
|
||||
demo2()
|
||||
|
||||
}
|
||||
|
||||
func demo2() {
|
||||
buf := `[settings]
|
||||
username=admin
|
||||
timeout=100
|
||||
`
|
||||
reader := inih.NewReader(c.Str(buf), c.Ulong(len(buf)))
|
||||
|
||||
println(&reader)
|
||||
println(reader.ParseError())
|
||||
sec := inih.Str("settings")
|
||||
name := inih.Str("timeout")
|
||||
value := reader.GetInteger(sec, name, 0)
|
||||
println("value:", value)
|
||||
}
|
||||
@@ -5,7 +5,7 @@ import (
|
||||
"github.com/goplus/llgo/cpp/inih"
|
||||
)
|
||||
|
||||
func demo1() {
|
||||
func main() {
|
||||
filename := c.Str("config.ini")
|
||||
|
||||
if inih.Parse(filename, func(user c.Pointer, section *c.Char, name *c.Char, value *c.Char) c.Int {
|
||||
|
||||
10
cpp/inih/_demo/inihreader/config.ini
Normal file
10
cpp/inih/_demo/inihreader/config.ini
Normal file
@@ -0,0 +1,10 @@
|
||||
; example.ini
|
||||
[owner]
|
||||
name = John Doe
|
||||
organization = Acme Widgets Inc.
|
||||
|
||||
[database]
|
||||
server = 192.0.2.62
|
||||
port = 143
|
||||
file = "payroll.dat"
|
||||
enabled = true
|
||||
42
cpp/inih/_demo/inihreader/reader_demo.go
Normal file
42
cpp/inih/_demo/inihreader/reader_demo.go
Normal file
@@ -0,0 +1,42 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"github.com/goplus/llgo/c"
|
||||
"github.com/goplus/llgo/cpp/inih"
|
||||
"github.com/goplus/llgo/cpp/std"
|
||||
)
|
||||
|
||||
func demoFromBuffer() {
|
||||
buf := `[settings]
|
||||
username=admin
|
||||
timeout=100
|
||||
`
|
||||
reader := inih.NewReader(c.Str(buf), uintptr(len(buf)))
|
||||
defer reader.Dispose()
|
||||
|
||||
println(reader.ParseError())
|
||||
sec := std.Str("settings")
|
||||
name := std.Str("timeout")
|
||||
value := reader.GetInteger(sec, name, 0)
|
||||
println("value:", value)
|
||||
}
|
||||
|
||||
func demoFromFile() {
|
||||
reader := inih.NewReaderFile(std.Str("config.ini"))
|
||||
defer reader.Dispose()
|
||||
|
||||
if ret := reader.ParseError(); ret != 0 {
|
||||
println("Error parsing config file:", ret)
|
||||
return
|
||||
}
|
||||
isDatabaseEnabled := reader.GetBoolean(std.Str("database"), std.Str("enabled"), false)
|
||||
port := reader.GetInteger(std.NewString("database"), std.Str("port"), 0)
|
||||
s := reader.GetString(std.Str("database"), std.Str("server"), std.Str("unknown"))
|
||||
println("s:", s.Str())
|
||||
println("isDatabaseEnabled:", isDatabaseEnabled, "port:", port)
|
||||
}
|
||||
|
||||
func main() {
|
||||
demoFromBuffer()
|
||||
demoFromFile()
|
||||
}
|
||||
13
cpp/inih/_wrap/reader.cpp
Normal file
13
cpp/inih/_wrap/reader.cpp
Normal file
@@ -0,0 +1,13 @@
|
||||
#include <INIReader.h>
|
||||
|
||||
extern "C" {
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
void INIReaderDispose(INIReader* r) {
|
||||
r->~INIReader();
|
||||
}
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
} // extern "C"
|
||||
@@ -1,3 +1,19 @@
|
||||
/*
|
||||
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package inih
|
||||
|
||||
import (
|
||||
@@ -7,7 +23,8 @@ import (
|
||||
)
|
||||
|
||||
const (
|
||||
LLGoPackage = "link: $(pkg-config --libs inih INIReader) -lc++; -linih -lINIReader"
|
||||
LLGoFiles = "$(pkg-config --cflags INIReader): _wrap/reader.cpp"
|
||||
LLGoPackage = "link: $(pkg-config --libs inih INIReader); -linih -lINIReader"
|
||||
)
|
||||
|
||||
//go:linkname Parse C.ini_parse
|
||||
|
||||
@@ -1,64 +0,0 @@
|
||||
package inih
|
||||
|
||||
import (
|
||||
"unsafe"
|
||||
|
||||
"github.com/goplus/llgo/c"
|
||||
)
|
||||
|
||||
// llgo:type C
|
||||
type Reader struct {
|
||||
Unused [24]byte
|
||||
}
|
||||
|
||||
type StdString struct {
|
||||
buf [24]byte
|
||||
}
|
||||
|
||||
type __long struct {
|
||||
__cap_ int
|
||||
__size_ int
|
||||
__data_ unsafe.Pointer
|
||||
}
|
||||
|
||||
func Str(s string) *StdString {
|
||||
var r StdString
|
||||
r.init(c.GoStringData(s), c.Int(len(s)))
|
||||
return &r
|
||||
}
|
||||
|
||||
func (r *StdString) String() string {
|
||||
if r.buf[0]&1 == 0 {
|
||||
return c.GoString((*c.Char)(unsafe.Pointer(&r.buf[1])))
|
||||
} else {
|
||||
v := *(*__long)(unsafe.Pointer(&r.buf[0]))
|
||||
return unsafe.String((*byte)(v.__data_), v.__size_)
|
||||
}
|
||||
}
|
||||
|
||||
// llgo:link (*StdString).init C._ZNSt3__112basic_stringIcNS_11char_traitsIcEENS_9allocatorIcEEE6__initEPKcm
|
||||
func (*StdString) init(s *c.Char, size c.Int) {}
|
||||
|
||||
//go:linkname NewReader C._ZN9INIReaderC1EPKcm
|
||||
func NewReader(fileName *c.Char, size c.Ulong) Reader
|
||||
|
||||
//go:linkname NewReaderFile C._ZN9INIReaderC1ERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEE
|
||||
func NewReaderFile(fileName *StdString) Reader
|
||||
|
||||
// llgo:link (*Reader).ParseError C._ZNK9INIReader10ParseErrorEv
|
||||
func (*Reader) ParseError() c.Int { return 0 }
|
||||
|
||||
// llgo:link (*Reader).GetInteger C._ZNK9INIReader10GetIntegerERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEES8_l
|
||||
func (*Reader) GetInteger(section *StdString, name *StdString, defaultValue c.Long) c.Long {
|
||||
return 0
|
||||
}
|
||||
|
||||
// llgo:link (*Reader).GetBoolean C._ZNK9INIReader10GetBooleanERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEES8_b
|
||||
func (*Reader) GetBoolean(section *StdString, name *StdString, defaultValue bool) bool {
|
||||
return false
|
||||
}
|
||||
|
||||
// llgo:link (*Reader).GetString C._ZNK9INIReader9GetStringERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEES8_S8_
|
||||
func (*Reader) GetString(section *StdString, name *StdString, defaultValue *StdString) StdString {
|
||||
return StdString{}
|
||||
}
|
||||
78
cpp/inih/reader.go
Normal file
78
cpp/inih/reader.go
Normal file
@@ -0,0 +1,78 @@
|
||||
/*
|
||||
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package inih
|
||||
|
||||
import (
|
||||
_ "unsafe"
|
||||
|
||||
"github.com/goplus/llgo/c"
|
||||
"github.com/goplus/llgo/cpp/std"
|
||||
)
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
// llgo:type C
|
||||
type Reader struct {
|
||||
Unused [24]byte
|
||||
}
|
||||
|
||||
// llgo:link (*Reader).InitFromBuffer C._ZN9INIReaderC1EPKcm
|
||||
func (r *Reader) InitFromBuffer(buffer *c.Char, bufferSize uintptr) {}
|
||||
|
||||
// llgo:link (*Reader).InitFromFile C._ZN9INIReaderC1ERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEE
|
||||
func (r *Reader) InitFromFile(fileName *std.String) {}
|
||||
|
||||
// llgo:link (*Reader).Dispose C.INIReaderDispose
|
||||
func (s *Reader) Dispose() {}
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
// NewReader creates a new INIReader instance.
|
||||
func NewReader(buffer *c.Char, bufferSize uintptr) (ret Reader) {
|
||||
ret.InitFromBuffer(buffer, bufferSize)
|
||||
return
|
||||
}
|
||||
|
||||
// NewReaderFile creates a new INIReader instance.
|
||||
func NewReaderFile(fileName *std.String) (ret Reader) {
|
||||
ret.InitFromFile(fileName)
|
||||
return
|
||||
}
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
// llgo:link (*Reader).ParseError C._ZNK9INIReader10ParseErrorEv
|
||||
func (*Reader) ParseError() c.Int { return 0 }
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
// llgo:link (*Reader).GetInteger C._ZNK9INIReader10GetIntegerERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEES8_l
|
||||
func (*Reader) GetInteger(section *std.String, name *std.String, defaultValue c.Long) c.Long {
|
||||
return 0
|
||||
}
|
||||
|
||||
// llgo:link (*Reader).GetBoolean C._ZNK9INIReader10GetBooleanERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEES8_b
|
||||
func (*Reader) GetBoolean(section *std.String, name *std.String, defaultValue bool) bool {
|
||||
return false
|
||||
}
|
||||
|
||||
// llgo:link (*Reader).GetString C._ZNK9INIReader9GetStringERKNSt3__112basic_stringIcNS0_11char_traitsIcEENS0_9allocatorIcEEEES8_S8_
|
||||
func (*Reader) GetString(section *std.String, name *std.String, defaultValue *std.String) (ret std.String) {
|
||||
return
|
||||
}
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
12
cpp/std/_demo/cpphello/cppstr.go
Normal file
12
cpp/std/_demo/cpphello/cppstr.go
Normal file
@@ -0,0 +1,12 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"github.com/goplus/llgo/c"
|
||||
"github.com/goplus/llgo/cpp/std"
|
||||
)
|
||||
|
||||
func main() {
|
||||
s := std.Str("Hello world\n")
|
||||
c.Printf(s.CStr())
|
||||
print(s.Str(), s.Size(), "\n")
|
||||
}
|
||||
43
cpp/std/_wrap/string.cpp
Normal file
43
cpp/std/_wrap/string.cpp
Normal file
@@ -0,0 +1,43 @@
|
||||
#include <string>
|
||||
|
||||
extern "C" {
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
void stdStringInitEmpty(std::string* s) {
|
||||
new(s) std::string();
|
||||
}
|
||||
|
||||
void stdStringInitFrom(std::string* s, std::string* v) {
|
||||
new(s) std::string(*v);
|
||||
}
|
||||
|
||||
void stdStringInitFromCStr(std::string* s, const char* cstr) {
|
||||
new(s) std::string(cstr);
|
||||
}
|
||||
|
||||
void stdStringInitFromCStrLen(std::string* s, const char* cstr, size_t len) {
|
||||
new(s) std::string(cstr, len);
|
||||
}
|
||||
|
||||
void stdStringDispose(std::string* s) {
|
||||
s->~basic_string();
|
||||
}
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
const char* stdStringCStr(const std::string* s) {
|
||||
return s->c_str();
|
||||
}
|
||||
|
||||
const char* stdStringData(const std::string* s) {
|
||||
return s->data();
|
||||
}
|
||||
|
||||
size_t stdStringSize(const std::string* s) {
|
||||
return s->size();
|
||||
}
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
} // extern "C"
|
||||
24
cpp/std/std.go
Normal file
24
cpp/std/std.go
Normal file
@@ -0,0 +1,24 @@
|
||||
/*
|
||||
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package std
|
||||
|
||||
const (
|
||||
LLGoFiles = "_wrap/string.cpp"
|
||||
LLGoPackage = "link: c++"
|
||||
)
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
121
cpp/std/string.go
Normal file
121
cpp/std/string.go
Normal file
@@ -0,0 +1,121 @@
|
||||
/*
|
||||
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package std
|
||||
|
||||
import (
|
||||
"unsafe"
|
||||
|
||||
"github.com/goplus/llgo/c"
|
||||
"github.com/goplus/llgo/c/bdwgc"
|
||||
)
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
// String represents a C++ std::string object.
|
||||
type String struct {
|
||||
Unused [24]byte
|
||||
}
|
||||
|
||||
// llgo:link (*String).InitEmpty C.stdStringInitEmpty
|
||||
func (s *String) InitEmpty() {}
|
||||
|
||||
// llgo:link (*String).InitFrom C.stdStringInitFrom
|
||||
func (s *String) InitFrom(v *String) {}
|
||||
|
||||
// llgo:link (*String).InitFromCStr C.stdStringInitFromCStr
|
||||
func (s *String) InitFromCStr(cstr *c.Char) {}
|
||||
|
||||
// llgo:link (*String).InitFromCStrLen C.stdStringInitFromCStrLen
|
||||
func (s *String) InitFromCStrLen(cstr *c.Char, n uintptr) {}
|
||||
|
||||
// llgo:link (*String).Dispose C.stdStringDispose
|
||||
func (s *String) Dispose() {}
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
func allocString() *String {
|
||||
ptr := bdwgc.Malloc(unsafe.Sizeof(String{}))
|
||||
bdwgc.RegisterFinalizer(ptr, func(obj, data c.Pointer) {
|
||||
(*String)(obj).Dispose()
|
||||
}, nil, nil, nil)
|
||||
return (*String)(ptr)
|
||||
}
|
||||
|
||||
// NewString creates a C++ std::string object.
|
||||
func NewString(v string) *String {
|
||||
ret := allocString()
|
||||
ret.InitFromCStrLen(c.GoStringData(v), uintptr(len(v)))
|
||||
return ret
|
||||
}
|
||||
|
||||
// NewStringEmpty creates an empty std::string object.
|
||||
func NewStringEmpty() *String {
|
||||
ret := allocString()
|
||||
ret.InitEmpty()
|
||||
return ret
|
||||
}
|
||||
|
||||
// NewStringFrom creates a copy of a C++ std::string object.
|
||||
func NewStringFrom(v *String) *String {
|
||||
ret := allocString()
|
||||
ret.InitFrom(v)
|
||||
return ret
|
||||
}
|
||||
|
||||
// NewStringFromCStr creates a C++ std::string object.
|
||||
func NewStringFromCStr(cstr *c.Char) *String {
|
||||
ret := allocString()
|
||||
ret.InitFromCStr(cstr)
|
||||
return ret
|
||||
}
|
||||
|
||||
// NewStringFromCStrLen creates a C++ std::string object.
|
||||
func NewStringFromCStrLen(cstr *c.Char, n uintptr) *String {
|
||||
ret := allocString()
|
||||
ret.InitFromCStrLen(cstr, n)
|
||||
return ret
|
||||
}
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
// Str returns a Go string (it doesn't clone data of the C++ std::string object).
|
||||
func (s *String) Str() string {
|
||||
return unsafe.String((*byte)(unsafe.Pointer(s.Data())), s.Size())
|
||||
}
|
||||
|
||||
// llgo:link (*String).CStr C.stdStringCStr
|
||||
func (s *String) CStr() *c.Char { return nil }
|
||||
|
||||
// llgo:link (*String).Data C.stdStringData
|
||||
func (s *String) Data() *c.Char { return nil }
|
||||
|
||||
// llgo:link (*String).Size C.stdStringSize
|
||||
func (s *String) Size() uintptr { return 0 }
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
|
||||
// GoString converts a C++ std::string object to a Go string.
|
||||
func GoString(v *String) string {
|
||||
return c.GoString(v.Data(), v.Size())
|
||||
}
|
||||
|
||||
// Str creates a constant C++ std::string object.
|
||||
func Str(v string) *String {
|
||||
return NewString(v) // TODO(xsw): optimize it
|
||||
}
|
||||
|
||||
// -----------------------------------------------------------------------------
|
||||
@@ -174,7 +174,7 @@ func Do(args []string, conf *Config) {
|
||||
patches := make(cl.Patches, len(altPkgPaths))
|
||||
altSSAPkgs(progSSA, patches, altPkgs[1:], verbose)
|
||||
|
||||
ctx := &context{progSSA, prog, dedup, patches, make(map[string]none), initial, mode}
|
||||
ctx := &context{llvm.New(""), progSSA, prog, dedup, patches, make(map[string]none), initial, mode}
|
||||
pkgs := buildAllPkgs(ctx, initial, verbose)
|
||||
|
||||
var llFiles []string
|
||||
@@ -189,7 +189,7 @@ func Do(args []string, conf *Config) {
|
||||
nErr := 0
|
||||
for _, pkg := range initial {
|
||||
if pkg.Name == "main" {
|
||||
nErr += linkMainPkg(pkg, pkgs, llFiles, conf, mode, verbose)
|
||||
nErr += linkMainPkg(ctx, pkg, pkgs, llFiles, conf, mode, verbose)
|
||||
}
|
||||
}
|
||||
if nErr > 0 {
|
||||
@@ -221,6 +221,7 @@ const (
|
||||
)
|
||||
|
||||
type context struct {
|
||||
env *llvm.Env
|
||||
progSSA *ssa.Program
|
||||
prog llssa.Program
|
||||
dedup packages.Deduper
|
||||
@@ -256,13 +257,14 @@ func buildAllPkgs(ctx *context, initial []*packages.Package, verbose bool) (pkgs
|
||||
case cl.PkgLinkIR, cl.PkgLinkExtern, cl.PkgPyModule:
|
||||
if len(pkg.GoFiles) > 0 {
|
||||
buildPkg(ctx, aPkg, verbose)
|
||||
pkg.ExportFile = " " + concatPkgLinkFiles(pkg, verbose) + " " + pkg.ExportFile
|
||||
pkg.ExportFile = " " + concatPkgLinkFiles(ctx, pkg, verbose) + " " + pkg.ExportFile
|
||||
} else {
|
||||
// panic("todo")
|
||||
// TODO(xsw): support packages out of llgo
|
||||
pkg.ExportFile = ""
|
||||
}
|
||||
if kind == cl.PkgLinkExtern { // need to be linked with external library
|
||||
if kind == cl.PkgLinkExtern {
|
||||
// need to be linked with external library
|
||||
// format: ';' separated alternative link methods. e.g.
|
||||
// link: $LLGO_LIB_PYTHON; $(pkg-config --libs python3-embed); -lpython3
|
||||
expd := ""
|
||||
@@ -305,7 +307,7 @@ func buildAllPkgs(ctx *context, initial []*packages.Package, verbose bool) (pkgs
|
||||
return
|
||||
}
|
||||
|
||||
func linkMainPkg(pkg *packages.Package, pkgs []*aPackage, llFiles []string, conf *Config, mode Mode, verbose bool) (nErr int) {
|
||||
func linkMainPkg(ctx *context, pkg *packages.Package, pkgs []*aPackage, llFiles []string, conf *Config, mode Mode, verbose bool) (nErr int) {
|
||||
pkgPath := pkg.PkgPath
|
||||
name := path.Base(pkgPath)
|
||||
app := conf.OutFile
|
||||
@@ -390,7 +392,7 @@ func linkMainPkg(pkg *packages.Package, pkgs []*aPackage, llFiles []string, conf
|
||||
if verbose {
|
||||
fmt.Fprintln(os.Stderr, "clang", args)
|
||||
}
|
||||
err := llvm.New("").Clang().Exec(args...)
|
||||
err := ctx.env.Clang().Exec(args...)
|
||||
check(err)
|
||||
|
||||
switch mode {
|
||||
@@ -589,6 +591,7 @@ func appendLinkFiles(args []string, file string) []string {
|
||||
if isSingleLinkFile(file) {
|
||||
return append(args, file)
|
||||
}
|
||||
// TODO(xsw): consider filename with spaces
|
||||
return append(args, strings.Split(file[1:], " ")...)
|
||||
}
|
||||
|
||||
@@ -596,11 +599,11 @@ func isSingleLinkFile(ret string) bool {
|
||||
return len(ret) > 0 && ret[0] != ' '
|
||||
}
|
||||
|
||||
func concatPkgLinkFiles(pkg *packages.Package, verbose bool) string {
|
||||
func concatPkgLinkFiles(ctx *context, pkg *packages.Package, verbose bool) string {
|
||||
var b strings.Builder
|
||||
var ret string
|
||||
var n int
|
||||
llgoPkgLinkFiles(pkg, func(linkFile string) {
|
||||
llgoPkgLinkFiles(ctx, pkg, func(linkFile string) {
|
||||
if n == 0 {
|
||||
ret = linkFile
|
||||
} else {
|
||||
@@ -618,32 +621,41 @@ func concatPkgLinkFiles(pkg *packages.Package, verbose bool) string {
|
||||
}
|
||||
|
||||
// const LLGoFiles = "file1; file2; ..."
|
||||
func llgoPkgLinkFiles(pkg *packages.Package, procFile func(linkFile string), verbose bool) {
|
||||
func llgoPkgLinkFiles(ctx *context, pkg *packages.Package, procFile func(linkFile string), verbose bool) {
|
||||
if o := pkg.Types.Scope().Lookup("LLGoFiles"); o != nil {
|
||||
val := o.(*types.Const).Val()
|
||||
if val.Kind() == constant.String {
|
||||
clFiles(constant.StringVal(val), pkg, procFile, verbose)
|
||||
clFiles(ctx, constant.StringVal(val), pkg, procFile, verbose)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// files = "file1; file2; ..."
|
||||
func clFiles(files string, pkg *packages.Package, procFile func(linkFile string), verbose bool) {
|
||||
// files = "$(pkg-config --cflags xxx): file1; file2; ..."
|
||||
func clFiles(ctx *context, files string, pkg *packages.Package, procFile func(linkFile string), verbose bool) {
|
||||
dir := filepath.Dir(pkg.GoFiles[0])
|
||||
expFile := pkg.ExportFile
|
||||
args := make([]string, 0, 16)
|
||||
if strings.HasPrefix(files, "$") { // has cflags
|
||||
if pos := strings.IndexByte(files, ':'); pos > 0 {
|
||||
cflags := env.ExpandEnv(files[:pos])
|
||||
files = files[pos+1:]
|
||||
args = append(args, strings.Split(cflags, " ")...)
|
||||
}
|
||||
}
|
||||
for _, file := range strings.Split(files, ";") {
|
||||
cFile := filepath.Join(dir, strings.TrimSpace(file))
|
||||
clFile(cFile, expFile, procFile, verbose)
|
||||
clFile(ctx, args, cFile, expFile, procFile, verbose)
|
||||
}
|
||||
}
|
||||
|
||||
func clFile(cFile, expFile string, procFile func(linkFile string), verbose bool) {
|
||||
func clFile(ctx *context, args []string, cFile, expFile string, procFile func(linkFile string), verbose bool) {
|
||||
llFile := expFile + filepath.Base(cFile) + ".ll"
|
||||
args := []string{"-emit-llvm", "-S", "-o", llFile, "-c", cFile}
|
||||
args = append(args, "-emit-llvm", "-S", "-o", llFile, "-c", cFile)
|
||||
if verbose {
|
||||
fmt.Fprintln(os.Stderr, "clang", args)
|
||||
}
|
||||
err := llvm.New("").Clang().Exec(args...)
|
||||
err := ctx.env.Clang().Exec(args...)
|
||||
check(err)
|
||||
procFile(llFile)
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user