 6e55234c38
			
		
	
	6e55234c38
	
	
	
		
			
			Add unit test to function GetCPUVariantFromArch Fix import issue on non-linux platforms Fix some style issue Signed-off-by: Tony Fang <nenghui.fang@gmail.com>
		
			
				
	
	
		
			162 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			162 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
|    Copyright The containerd Authors.
 | |
| 
 | |
|    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 platforms
 | |
| 
 | |
| import (
 | |
| 	"bufio"
 | |
| 	"bytes"
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"runtime"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/containerd/containerd/errdefs"
 | |
| 	"golang.org/x/sys/unix"
 | |
| )
 | |
| 
 | |
| // getMachineArch retrieves the machine architecture through system call
 | |
| func getMachineArch() (string, error) {
 | |
| 	var uname unix.Utsname
 | |
| 	err := unix.Uname(&uname)
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	arch := string(uname.Machine[:bytes.IndexByte(uname.Machine[:], 0)])
 | |
| 
 | |
| 	return arch, nil
 | |
| }
 | |
| 
 | |
| // For Linux, the kernel has already detected the ABI, ISA and Features.
 | |
| // So we don't need to access the ARM registers to detect platform information
 | |
| // by ourselves. We can just parse these information from /proc/cpuinfo
 | |
| func getCPUInfo(pattern string) (info string, err error) {
 | |
| 
 | |
| 	cpuinfo, err := os.Open("/proc/cpuinfo")
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	defer cpuinfo.Close()
 | |
| 
 | |
| 	// Start to Parse the Cpuinfo line by line. For SMP SoC, we parse
 | |
| 	// the first core is enough.
 | |
| 	scanner := bufio.NewScanner(cpuinfo)
 | |
| 	for scanner.Scan() {
 | |
| 		newline := scanner.Text()
 | |
| 		list := strings.Split(newline, ":")
 | |
| 
 | |
| 		if len(list) > 1 && strings.EqualFold(strings.TrimSpace(list[0]), pattern) {
 | |
| 			return strings.TrimSpace(list[1]), nil
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Check whether the scanner encountered errors
 | |
| 	err = scanner.Err()
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 
 | |
| 	return "", fmt.Errorf("getCPUInfo for pattern %s: %w", pattern, errdefs.ErrNotFound)
 | |
| }
 | |
| 
 | |
| // getCPUVariantFromArch get CPU variant from arch through a system call
 | |
| func getCPUVariantFromArch(arch string) (string, error) {
 | |
| 
 | |
| 	var variant string
 | |
| 
 | |
| 	arch = strings.ToLower(arch)
 | |
| 
 | |
| 	if arch == "aarch64" {
 | |
| 		variant = "8"
 | |
| 	} else if arch[0:4] == "armv" && len(arch) >= 5 {
 | |
| 		//Valid arch format is in form of armvXx
 | |
| 		switch arch[3:5] {
 | |
| 		case "v8":
 | |
| 			variant = "8"
 | |
| 		case "v7":
 | |
| 			variant = "7"
 | |
| 		case "v6":
 | |
| 			variant = "6"
 | |
| 		case "v5":
 | |
| 			variant = "5"
 | |
| 		case "v4":
 | |
| 			variant = "4"
 | |
| 		case "v3":
 | |
| 			variant = "3"
 | |
| 		default:
 | |
| 			variant = "unknown"
 | |
| 		}
 | |
| 	} else {
 | |
| 		return "", fmt.Errorf("getCPUVariantFromArch invalid arch: %s, %w", arch, errdefs.ErrInvalidArgument)
 | |
| 	}
 | |
| 	return variant, nil
 | |
| }
 | |
| 
 | |
| // getCPUVariant returns cpu variant for ARM
 | |
| // We first try reading "Cpu architecture" field from /proc/cpuinfo
 | |
| // If we can't find it, then fall back using a system call
 | |
| // This is to cover running ARM in emulated environment on x86 host as this field in /proc/cpuinfo
 | |
| // was not present.
 | |
| func getCPUVariant() (string, error) {
 | |
| 
 | |
| 	variant, err := getCPUInfo("Cpu architecture")
 | |
| 	if err != nil {
 | |
| 		if errdefs.IsNotFound(err) {
 | |
| 			//Let's try getting CPU variant from machine architecture
 | |
| 			arch, err := getMachineArch()
 | |
| 			if err != nil {
 | |
| 				return "", fmt.Errorf("failure getting machine architecture: %v", err)
 | |
| 			}
 | |
| 
 | |
| 			variant, err = getCPUVariantFromArch(arch)
 | |
| 			if err != nil {
 | |
| 				return "", fmt.Errorf("failure getting CPU variant from machine architecture: %v", err)
 | |
| 			}
 | |
| 		} else {
 | |
| 			return "", fmt.Errorf("failure getting CPU variant: %v", err)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// handle edge case for Raspberry Pi ARMv6 devices (which due to a kernel quirk, report "CPU architecture: 7")
 | |
| 	// https://www.raspberrypi.org/forums/viewtopic.php?t=12614
 | |
| 	if runtime.GOARCH == "arm" && variant == "7" {
 | |
| 		model, err := getCPUInfo("model name")
 | |
| 		if err == nil && strings.HasPrefix(strings.ToLower(model), "armv6-compatible") {
 | |
| 			variant = "6"
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	switch strings.ToLower(variant) {
 | |
| 	case "8", "aarch64":
 | |
| 		variant = "v8"
 | |
| 	case "7", "7m", "?(12)", "?(13)", "?(14)", "?(15)", "?(16)", "?(17)":
 | |
| 		variant = "v7"
 | |
| 	case "6", "6tej":
 | |
| 		variant = "v6"
 | |
| 	case "5", "5t", "5te", "5tej":
 | |
| 		variant = "v5"
 | |
| 	case "4", "4t":
 | |
| 		variant = "v4"
 | |
| 	case "3":
 | |
| 		variant = "v3"
 | |
| 	default:
 | |
| 		variant = "unknown"
 | |
| 	}
 | |
| 
 | |
| 	return variant, nil
 | |
| }
 |