437 lines
12 KiB
Go
437 lines
12 KiB
Go
// Copyright 2019 The UNICORE Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file
|
|
//
|
|
// Author: Gaulthier Gain <gaulthier.gain@uliege.be>
|
|
|
|
package buildtool
|
|
|
|
import (
|
|
"bufio"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
|
|
u "tools/srcs/common"
|
|
)
|
|
|
|
// ---------------------------Create Patches Folder-----------------------------
|
|
|
|
func createPatchFolder(appFolder string) (*string, error) {
|
|
|
|
patchesFolder := appFolder + "patches" + u.SEP
|
|
if _, err := u.CreateFolder(patchesFolder); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &patchesFolder, nil
|
|
}
|
|
|
|
// ---------------------------Create Include Folder-----------------------------
|
|
|
|
func createIncludeFolder(appFolder string) (*string, error) {
|
|
|
|
includeFolder := appFolder + u.INCLUDEFOLDER
|
|
if _, err := u.CreateFolder(includeFolder); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &includeFolder, nil
|
|
}
|
|
|
|
// ----------------------------Set Workspace Folders-----------------------------
|
|
func setWorkspaceFolder(workspacePath string) error {
|
|
|
|
_, err := u.CreateFolder(workspacePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func setUnikraftSubFolders(workspaceFolder string) error {
|
|
|
|
unikraftFolder := workspaceFolder + u.UNIKRAFTFOLDER
|
|
u.PrintInfo("Managing Unikraft main folder with apps and libs subfolders")
|
|
|
|
// Create 'apps' and 'libs' subfolders
|
|
if _, err := u.CreateFolder(workspaceFolder + u.APPSFOLDER); err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, err := u.CreateFolder(workspaceFolder + u.LIBSFOLDER); err != nil {
|
|
return err
|
|
}
|
|
|
|
if _, err := os.Stat(unikraftFolder); os.IsNotExist(err) {
|
|
url := "https://github.com/unikraft/unikraft.git"
|
|
// Download git repo of unikraft
|
|
if _, _, err := u.GitCloneRepository(url, workspaceFolder, true); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
// Use staging branch
|
|
if _, _, err := u.GitBranchStaging(unikraftFolder, false); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// ---------------------------Check UNIKRAFT Folder-----------------------------
|
|
|
|
func containsUnikraftFolders(files []os.FileInfo) bool {
|
|
|
|
if len(files) == 0 {
|
|
return false
|
|
}
|
|
|
|
m := make(map[string]bool)
|
|
m[u.APPSFOLDER], m[u.LIBSFOLDER], m[u.UNIKRAFTFOLDER] = false, false, false
|
|
|
|
var folderName string
|
|
for _, f := range files {
|
|
folderName = f.Name() + u.SEP
|
|
if _, ok := m[folderName]; ok {
|
|
m[folderName] = true
|
|
}
|
|
}
|
|
|
|
return m[u.APPSFOLDER] == true && m[u.LIBSFOLDER] && m[u.UNIKRAFTFOLDER]
|
|
}
|
|
|
|
// ---------------------------UNIKRAFT APP FOLDER-------------------------------
|
|
|
|
func createUnikraftApp(programName, workspacePath string) (*string, error) {
|
|
|
|
var appFolder string
|
|
if workspacePath[len(workspacePath)-1] != os.PathSeparator {
|
|
appFolder = workspacePath + u.SEP + u.APPSFOLDER + programName + u.SEP
|
|
} else {
|
|
appFolder = workspacePath + u.APPSFOLDER + programName + u.SEP
|
|
}
|
|
|
|
created, err := u.CreateFolder(appFolder)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !created {
|
|
u.PrintWarning(appFolder + " already exists.")
|
|
appFolder = handleCreationApp(appFolder)
|
|
if _, err := u.CreateFolder(appFolder); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return &appFolder, nil
|
|
}
|
|
|
|
// -----------------------------Create App folder-------------------------------
|
|
|
|
func handleCreationApp(appFolder string) string {
|
|
fmt.Println("Make your choice:\n1: Copy and overwrite files\n2: " +
|
|
"Enter manually the name of the folder\n3: exit program")
|
|
var input int
|
|
for {
|
|
fmt.Print("Please enter your choice (0 to exit): ")
|
|
if _, err := fmt.Scanf("%d", &input); err != nil {
|
|
u.PrintWarning("Choice must be numeric! Try again")
|
|
} else {
|
|
switch input {
|
|
case 1:
|
|
return appFolder
|
|
case 2:
|
|
fmt.Print("Enter text: ")
|
|
reader := bufio.NewReader(os.Stdin)
|
|
text, _ := reader.ReadString('\n')
|
|
appFolder = strings.Split(text, "\n")[0] + u.SEP
|
|
return appFolder
|
|
case 3:
|
|
os.Exit(1)
|
|
default:
|
|
u.PrintWarning("Invalid input! Try again")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// -------------------------MOVE FILES TO APP FOLDER----------------------------
|
|
|
|
var srcLanguages = map[string]int{
|
|
".c": 0,
|
|
".cpp": 0,
|
|
".cc": 0,
|
|
".S": 0,
|
|
".s": 0,
|
|
".asm": 0,
|
|
".py": 0,
|
|
".go": 0,
|
|
}
|
|
|
|
func filterSourcesFiles(sourceFiles []string) []string {
|
|
filterSrcFiles := make([]string, 0)
|
|
for _, file := range sourceFiles {
|
|
if !strings.Contains(file, "copy") &&
|
|
!strings.Contains(file, "test") &&
|
|
!strings.Contains(file, "unit") {
|
|
filterSrcFiles = append(filterSrcFiles, file)
|
|
}
|
|
|
|
}
|
|
return filterSrcFiles
|
|
}
|
|
|
|
// addAndApplyPatchFiles copies all the user-provided patch files to the unikernel directory,
|
|
// conforms them to the unikernel directory format so that all paths in the patch files are paths
|
|
// to source files located in the unikernel folder and applies the patches.
|
|
//
|
|
// It returns an error if any, otherwise it returns nil.
|
|
func addAndApplyPatchFiles(patchPath string, patchFolder, appFolder string) error {
|
|
|
|
// Copy and conform patch files
|
|
err := filepath.Walk(patchPath, func(filePath string, info os.FileInfo,
|
|
err error) error {
|
|
if !info.IsDir() {
|
|
extension := filepath.Ext(info.Name())
|
|
if extension == ".patch" {
|
|
if err = conformPatchFile(filePath, patchFolder+info.Name()); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Initialise git repo to be able to apply patches
|
|
_, _, _ = u.ExecuteRunCmd("git", appFolder, true, "init")
|
|
_, _, _ = u.ExecuteRunCmd("git", appFolder, true, "add", ".")
|
|
_, _, _ = u.ExecuteRunCmd("git", appFolder, true, "commit", "-m", "first commit")
|
|
|
|
// Apply patches
|
|
err = filepath.Walk(patchPath, func(filePath string, info os.FileInfo,
|
|
err error) error {
|
|
if !info.IsDir() {
|
|
_, _, _ = u.ExecuteRunCmd("git", appFolder, true, "am", patchFolder+info.Name())
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// conformPatchPath conforms a path in a user-provided patch file to the unikernel directory format
|
|
// so that this path describes a source file located in the unikernel folder.
|
|
func conformPatchPath(match *[]string, index int) {
|
|
extension := filepath.Ext((*match)[index])
|
|
if extension == ".h" || extension == ".hpp" || extension == ".hcc" {
|
|
(*match)[index] = "include" + u.SEP + filepath.Base((*match)[index])
|
|
} else if extension == ".c" || extension == ".cpp" || extension == ".cc" {
|
|
(*match)[index] = filepath.Base((*match)[index])
|
|
} else {
|
|
u.PrintWarning("Unsupported extension for file: " +
|
|
filepath.Base((*match)[index]))
|
|
}
|
|
}
|
|
|
|
// conformPatchFile copies all the user-provided patch files to the unikernel directory and
|
|
// conforms them to the unikernel directory format so that all paths in the patch files are paths
|
|
// to source files located in the unikernel folder.
|
|
//
|
|
// It returns an error if any, otherwise it returns nil.
|
|
func conformPatchFile(patchPath, newPatchPath string) error {
|
|
|
|
patchLines, err := u.ReadLinesFile(patchPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Find paths in patch file using regexp
|
|
var re1 = regexp.MustCompile(`( )(.*)( \| )(.*)`)
|
|
var re2 = regexp.MustCompile(`(diff --git )(a/)?(.*)( )(b/)?(.*)`)
|
|
var re3 = regexp.MustCompile(`(--- )(a/)?(.*)`)
|
|
var re4 = regexp.MustCompile(`(\+\+\+ )(b/)?(.*)`)
|
|
|
|
for lineIndex := range patchLines {
|
|
|
|
// All paths to files to be modified by the patch are listed under "---"
|
|
if patchLines[lineIndex] == "---\n" {
|
|
lineIndex++
|
|
for ; !strings.Contains(patchLines[lineIndex], "changed"); lineIndex++ {
|
|
for _, match := range re1.FindAllStringSubmatch(patchLines[lineIndex], -1) {
|
|
conformPatchPath(&match, 2)
|
|
patchLines[lineIndex] = strings.Join(match[1:], "") + "\n"
|
|
}
|
|
}
|
|
}
|
|
|
|
// All diff lines contain paths to files to be modified by the patch
|
|
if len(patchLines[lineIndex]) > 10 && patchLines[lineIndex][:10] == "diff --git" {
|
|
for _, match := range re2.FindAllStringSubmatch(patchLines[lineIndex], -1) {
|
|
conformPatchPath(&match, 3)
|
|
conformPatchPath(&match, 6)
|
|
patchLines[lineIndex] = strings.Join(match[1:], "") + "\n"
|
|
}
|
|
|
|
// Same observation for the two lines following the index line
|
|
for _, match := range re3.FindAllStringSubmatch(patchLines[lineIndex+2], -1) {
|
|
conformPatchPath(&match, 3)
|
|
patchLines[lineIndex+2] = strings.Join(match[1:], "") + "\n"
|
|
}
|
|
for _, match := range re4.FindAllStringSubmatch(patchLines[lineIndex+3], -1) {
|
|
conformPatchPath(&match, 3)
|
|
patchLines[lineIndex+3] = strings.Join(match[1:], "") + "\n"
|
|
}
|
|
}
|
|
}
|
|
|
|
// Write the modified content to a file in the unikernel folder
|
|
err = u.WriteToFile(newPatchPath, []byte(strings.Join(patchLines, "")))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// conformIncludeDirectives conforms all the user-defined include directives of all C/C++ source
|
|
// files to the unikernel directory format so that all these directives are paths to header files
|
|
// located in the include folder of the unikernel directory.
|
|
//
|
|
// It returns an error if any, otherwise it returns nil.
|
|
func conformIncludeDirectives(sourcePath string) error {
|
|
|
|
err := filepath.Walk(sourcePath, func(path string, info os.FileInfo,
|
|
err error) error {
|
|
if !info.IsDir() {
|
|
extension := filepath.Ext(info.Name())
|
|
if extension == ".h" || extension == ".hpp" || extension == ".hcc" {
|
|
if err = conformFile(path, true); err != nil {
|
|
return err
|
|
}
|
|
} else if extension == ".c" || extension == ".cpp" || extension == ".cc" {
|
|
if err = conformFile(path, false); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
u.PrintWarning("Unsupported extension for file: " + info.Name())
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// conformFile conforms all the user-defined include directives of a C/C++ source file to the
|
|
// unikernel directory format so that all these directives are paths to header files located in the
|
|
// include folder.
|
|
//
|
|
// It returns an error if any, otherwise it returns nil.
|
|
func conformFile(filePath string, isHeader bool) (err error) {
|
|
|
|
fileLines, err := u.ReadLinesFile(filePath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Find user-defined include directives using regexp
|
|
var re = regexp.MustCompile(`(.*)(#include)(.*)(")(.*)(")(.*)`)
|
|
|
|
for lineIndex := range fileLines {
|
|
for _, match := range re.FindAllStringSubmatch(fileLines[lineIndex], -1) {
|
|
|
|
// Replace the path by (include/ +) its last element
|
|
for i := 1; i < len(match); i++ {
|
|
if match[i] == "\"" {
|
|
if isHeader {
|
|
match[i+1] = filepath.Base(match[i+1])
|
|
} else {
|
|
match[i+1] = "include" + u.SEP + filepath.Base(match[i+1])
|
|
}
|
|
fileLines[lineIndex] = strings.Join(match[1:], "") + "\n"
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Write the modified content to a file in the unikernel folder
|
|
err = u.WriteToFile(filePath, []byte(strings.Join(fileLines, "")))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func processSourceFiles(sourcesPath, appFolder, includeFolder string,
|
|
sourceFiles, includesFiles []string) ([]string, error) {
|
|
|
|
err := filepath.Walk(sourcesPath, func(path string, info os.FileInfo,
|
|
err error) error {
|
|
if !info.IsDir() {
|
|
extension := filepath.Ext(info.Name())
|
|
if _, ok := srcLanguages[extension]; ok {
|
|
// Add source files to sourceFiles list
|
|
sourceFiles = append(sourceFiles, info.Name())
|
|
|
|
// Count the number of extension
|
|
srcLanguages[extension] += 1
|
|
|
|
// Copy source files to the appFolder
|
|
if err = u.CopyFileContents(path, appFolder+info.Name()); err != nil {
|
|
return err
|
|
}
|
|
} else if extension == ".h" || extension == ".hpp" || extension == ".hcc" {
|
|
// Add source files to includesFiles list
|
|
includesFiles = append(includesFiles, info.Name())
|
|
|
|
// Copy header files to the INCLUDEFOLDER
|
|
if err = u.CopyFileContents(path, includeFolder+info.Name()); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
u.PrintWarning("Unsupported extension for file: " + info.Name())
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// If no source file, exit the program
|
|
if len(sourceFiles) == 0 {
|
|
return nil, errors.New("unable to find source files")
|
|
}
|
|
|
|
return sourceFiles, nil
|
|
}
|
|
|
|
func languageUsed() string {
|
|
|
|
max := -1
|
|
var mostUsedFiles string
|
|
for key, value := range srcLanguages {
|
|
if max < value {
|
|
max = value
|
|
mostUsedFiles = key
|
|
}
|
|
}
|
|
|
|
return mostUsedFiles
|
|
}
|