mirror of
https://github.com/token2/snapd.git
synced 2026-03-13 11:15:47 -07:00
368 lines
12 KiB
Go
368 lines
12 KiB
Go
// -*- Mode: Go; indent-tabs-mode: t -*-
|
|
|
|
/*
|
|
* Copyright (C) 2014-2015 Canonical Ltd
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 3 as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
package bootloader_test
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
. "gopkg.in/check.v1"
|
|
|
|
"github.com/snapcore/snapd/bootloader"
|
|
"github.com/snapcore/snapd/bootloader/assets"
|
|
"github.com/snapcore/snapd/bootloader/bootloadertest"
|
|
"github.com/snapcore/snapd/dirs"
|
|
"github.com/snapcore/snapd/snap"
|
|
"github.com/snapcore/snapd/testutil"
|
|
)
|
|
|
|
// Hook up check.v1 into the "go test" runner
|
|
func Test(t *testing.T) { TestingT(t) }
|
|
|
|
const packageKernel = `
|
|
name: ubuntu-kernel
|
|
version: 4.0-1
|
|
type: kernel
|
|
vendor: Someone
|
|
`
|
|
|
|
type baseBootenvTestSuite struct {
|
|
testutil.BaseTest
|
|
|
|
rootdir string
|
|
}
|
|
|
|
func (s *baseBootenvTestSuite) SetUpTest(c *C) {
|
|
s.BaseTest.SetUpTest(c)
|
|
s.AddCleanup(snap.MockSanitizePlugsSlots(func(snapInfo *snap.Info) {}))
|
|
s.rootdir = c.MkDir()
|
|
dirs.SetRootDir(s.rootdir)
|
|
s.AddCleanup(func() { dirs.SetRootDir("") })
|
|
}
|
|
|
|
type bootenvTestSuite struct {
|
|
baseBootenvTestSuite
|
|
|
|
b *bootloadertest.MockBootloader
|
|
}
|
|
|
|
var _ = Suite(&bootenvTestSuite{})
|
|
|
|
func (s *bootenvTestSuite) SetUpTest(c *C) {
|
|
s.baseBootenvTestSuite.SetUpTest(c)
|
|
|
|
s.b = bootloadertest.Mock("mocky", c.MkDir())
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestForceBootloader(c *C) {
|
|
bootloader.Force(s.b)
|
|
defer bootloader.Force(nil)
|
|
|
|
got, err := bootloader.Find("", nil)
|
|
c.Assert(err, IsNil)
|
|
c.Check(got, Equals, s.b)
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestForceBootloaderError(c *C) {
|
|
myErr := errors.New("zap")
|
|
bootloader.ForceError(myErr)
|
|
defer bootloader.ForceError(nil)
|
|
|
|
got, err := bootloader.Find("", nil)
|
|
c.Assert(err, Equals, myErr)
|
|
c.Check(got, IsNil)
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestInstallBootloaderConfigNoConfig(c *C) {
|
|
err := bootloader.InstallBootConfig(c.MkDir(), s.rootdir, nil)
|
|
c.Assert(err, ErrorMatches, `cannot find boot config in.*`)
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestInstallBootloaderConfigFromGadget(c *C) {
|
|
for _, t := range []struct {
|
|
name string
|
|
gadgetFile, sysFile string
|
|
gadgetFileContent []byte
|
|
opts *bootloader.Options
|
|
}{
|
|
{name: "grub", gadgetFile: "grub.conf", sysFile: "/boot/grub/grub.cfg"},
|
|
// traditional uboot.env - the uboot.env file needs to be non-empty
|
|
{name: "uboot.env", gadgetFile: "uboot.conf", sysFile: "/boot/uboot/uboot.env", gadgetFileContent: []byte{1}},
|
|
// boot.scr in place of uboot.env means we create the boot.sel file
|
|
{
|
|
name: "uboot boot.scr",
|
|
gadgetFile: "uboot.conf",
|
|
sysFile: "/uboot/ubuntu/boot.sel",
|
|
opts: &bootloader.Options{Role: bootloader.RoleRecovery},
|
|
},
|
|
{name: "androidboot", gadgetFile: "androidboot.conf", sysFile: "/boot/androidboot/androidboot.env"},
|
|
{name: "lk", gadgetFile: "lk.conf", sysFile: "/boot/lk/snapbootsel.bin", opts: &bootloader.Options{PrepareImageTime: true}},
|
|
{
|
|
name: "piboot",
|
|
gadgetFile: "piboot.conf",
|
|
sysFile: "/boot/piboot/piboot.conf",
|
|
},
|
|
} {
|
|
mockGadgetDir := c.MkDir()
|
|
rootDir := c.MkDir()
|
|
err := ioutil.WriteFile(filepath.Join(mockGadgetDir, t.gadgetFile), t.gadgetFileContent, 0644)
|
|
c.Assert(err, IsNil)
|
|
err = bootloader.InstallBootConfig(mockGadgetDir, rootDir, t.opts)
|
|
c.Assert(err, IsNil, Commentf("installing boot config for %s", t.name))
|
|
fn := filepath.Join(rootDir, t.sysFile)
|
|
c.Assert(fn, testutil.FilePresent, Commentf("boot config missing for %s at %s", t.name, t.sysFile))
|
|
}
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestInstallBootloaderConfigFromAssets(c *C) {
|
|
recoveryOpts := &bootloader.Options{
|
|
Role: bootloader.RoleRecovery,
|
|
}
|
|
systemBootOpts := &bootloader.Options{
|
|
Role: bootloader.RoleRunMode,
|
|
}
|
|
defaultRecoveryGrubAsset := assets.Internal("grub-recovery.cfg")
|
|
c.Assert(defaultRecoveryGrubAsset, NotNil)
|
|
defaultGrubAsset := assets.Internal("grub.cfg")
|
|
c.Assert(defaultGrubAsset, NotNil)
|
|
|
|
for _, t := range []struct {
|
|
name string
|
|
gadgetFile, sysFile string
|
|
gadgetFileContent []byte
|
|
sysFileContent []byte
|
|
assetContent []byte
|
|
assetName string
|
|
err string
|
|
opts *bootloader.Options
|
|
}{
|
|
{
|
|
name: "recovery grub",
|
|
opts: recoveryOpts,
|
|
gadgetFile: "grub.conf",
|
|
// empty file in the gadget
|
|
gadgetFileContent: nil,
|
|
sysFile: "/EFI/ubuntu/grub.cfg",
|
|
assetName: "grub-recovery.cfg",
|
|
assetContent: []byte("hello assets"),
|
|
// boot config from assets
|
|
sysFileContent: []byte("hello assets"),
|
|
}, {
|
|
name: "recovery grub with non empty gadget file",
|
|
opts: recoveryOpts,
|
|
gadgetFile: "grub.conf",
|
|
gadgetFileContent: []byte("not so empty"),
|
|
sysFile: "/EFI/ubuntu/grub.cfg",
|
|
assetName: "grub-recovery.cfg",
|
|
assetContent: []byte("hello assets"),
|
|
// boot config from assets
|
|
sysFileContent: []byte("hello assets"),
|
|
}, {
|
|
name: "recovery grub with default asset",
|
|
opts: recoveryOpts,
|
|
gadgetFile: "grub.conf",
|
|
// empty file in the gadget
|
|
gadgetFileContent: nil,
|
|
sysFile: "/EFI/ubuntu/grub.cfg",
|
|
sysFileContent: defaultRecoveryGrubAsset,
|
|
}, {
|
|
name: "recovery grub missing asset",
|
|
opts: recoveryOpts,
|
|
gadgetFile: "grub.conf",
|
|
// empty file in the gadget
|
|
gadgetFileContent: nil,
|
|
sysFile: "/EFI/ubuntu/grub.cfg",
|
|
assetName: "grub-recovery.cfg",
|
|
// no asset content
|
|
err: `internal error: no boot asset for "grub-recovery.cfg"`,
|
|
}, {
|
|
name: "system-boot grub",
|
|
opts: systemBootOpts,
|
|
gadgetFile: "grub.conf",
|
|
// empty file in the gadget
|
|
gadgetFileContent: nil,
|
|
sysFile: "/EFI/ubuntu/grub.cfg",
|
|
assetName: "grub.cfg",
|
|
assetContent: []byte("hello assets"),
|
|
sysFileContent: []byte("hello assets"),
|
|
}, {
|
|
name: "system-boot grub with default asset",
|
|
opts: systemBootOpts,
|
|
gadgetFile: "grub.conf",
|
|
// empty file in the gadget
|
|
gadgetFileContent: nil,
|
|
sysFile: "/EFI/ubuntu/grub.cfg",
|
|
sysFileContent: defaultGrubAsset,
|
|
},
|
|
} {
|
|
mockGadgetDir := c.MkDir()
|
|
rootDir := c.MkDir()
|
|
fn := filepath.Join(rootDir, t.sysFile)
|
|
err := ioutil.WriteFile(filepath.Join(mockGadgetDir, t.gadgetFile), t.gadgetFileContent, 0644)
|
|
c.Assert(err, IsNil)
|
|
var restoreAsset func()
|
|
if t.assetName != "" {
|
|
restoreAsset = assets.MockInternal(t.assetName, t.assetContent)
|
|
}
|
|
err = bootloader.InstallBootConfig(mockGadgetDir, rootDir, t.opts)
|
|
if t.err == "" {
|
|
c.Assert(err, IsNil, Commentf("installing boot config for %s", t.name))
|
|
// mocked asset content
|
|
c.Assert(fn, testutil.FileEquals, string(t.sysFileContent))
|
|
} else {
|
|
c.Assert(err, ErrorMatches, t.err)
|
|
c.Assert(fn, testutil.FileAbsent)
|
|
}
|
|
if restoreAsset != nil {
|
|
restoreAsset()
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestBootloaderFindPresentNonNilError(c *C) {
|
|
rootdir := c.MkDir()
|
|
// add a mock bootloader to the list of bootloaders that Find() uses
|
|
mockBl := bootloadertest.Mock("mock", rootdir)
|
|
restore := bootloader.MockAddBootloaderToFind(func(dir string, opts *bootloader.Options) bootloader.Bootloader {
|
|
c.Assert(dir, Equals, rootdir)
|
|
return mockBl
|
|
})
|
|
defer restore()
|
|
|
|
// make us find our bootloader
|
|
mockBl.MockedPresent = true
|
|
|
|
bl, err := bootloader.Find(rootdir, nil)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(bl, NotNil)
|
|
c.Assert(bl.Name(), Equals, "mock")
|
|
c.Assert(bl, DeepEquals, mockBl)
|
|
|
|
// now make finding our bootloader a fatal error, this time we will get the
|
|
// error back
|
|
mockBl.PresentErr = fmt.Errorf("boom")
|
|
_, err = bootloader.Find(rootdir, nil)
|
|
c.Assert(err, ErrorMatches, "bootloader \"mock\" found but not usable: boom")
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestBootloaderFindBadOptions(c *C) {
|
|
_, err := bootloader.Find("", &bootloader.Options{
|
|
PrepareImageTime: true,
|
|
Role: bootloader.RoleRunMode,
|
|
})
|
|
c.Assert(err, ErrorMatches, "internal error: cannot use run mode bootloader at prepare-image time")
|
|
|
|
_, err = bootloader.Find("", &bootloader.Options{
|
|
NoSlashBoot: true,
|
|
Role: bootloader.RoleSole,
|
|
})
|
|
c.Assert(err, ErrorMatches, "internal error: bootloader.RoleSole doesn't expect NoSlashBoot set")
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestBootloaderFind(c *C) {
|
|
for _, tc := range []struct {
|
|
name string
|
|
sysFile string
|
|
opts *bootloader.Options
|
|
expName string
|
|
}{
|
|
{name: "grub", sysFile: "/boot/grub/grub.cfg", expName: "grub"},
|
|
{
|
|
// native run partition layout
|
|
name: "grub", sysFile: "/EFI/ubuntu/grub.cfg",
|
|
opts: &bootloader.Options{Role: bootloader.RoleRunMode, NoSlashBoot: true},
|
|
expName: "grub",
|
|
},
|
|
{
|
|
// recovery layout
|
|
name: "grub", sysFile: "/EFI/ubuntu/grub.cfg",
|
|
opts: &bootloader.Options{Role: bootloader.RoleRecovery},
|
|
expName: "grub",
|
|
},
|
|
|
|
// traditional uboot.env - the uboot.env file needs to be non-empty
|
|
{name: "uboot.env", sysFile: "/boot/uboot/uboot.env", expName: "uboot"},
|
|
// boot.sel uboot variant
|
|
{
|
|
name: "uboot boot.scr",
|
|
sysFile: "/uboot/ubuntu/boot.sel",
|
|
opts: &bootloader.Options{Role: bootloader.RoleRunMode, NoSlashBoot: true},
|
|
expName: "uboot",
|
|
},
|
|
{name: "androidboot", sysFile: "/boot/androidboot/androidboot.env", expName: "androidboot"},
|
|
// lk is detected differently based on runtime/prepare-image
|
|
{name: "lk", sysFile: "/dev/disk/by-partlabel/snapbootsel", expName: "lk"},
|
|
{
|
|
name: "lk", sysFile: "/boot/lk/snapbootsel.bin",
|
|
expName: "lk", opts: &bootloader.Options{PrepareImageTime: true},
|
|
},
|
|
} {
|
|
c.Logf("tc: %v", tc.name)
|
|
rootDir := c.MkDir()
|
|
err := os.MkdirAll(filepath.Join(rootDir, filepath.Dir(tc.sysFile)), 0755)
|
|
c.Assert(err, IsNil)
|
|
err = ioutil.WriteFile(filepath.Join(rootDir, tc.sysFile), nil, 0644)
|
|
c.Assert(err, IsNil)
|
|
bl, err := bootloader.Find(rootDir, tc.opts)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(bl, NotNil)
|
|
c.Check(bl.Name(), Equals, tc.expName)
|
|
}
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestBootloaderForGadget(c *C) {
|
|
for _, tc := range []struct {
|
|
name string
|
|
gadgetFile string
|
|
opts *bootloader.Options
|
|
expName string
|
|
}{
|
|
{name: "grub", gadgetFile: "grub.conf", expName: "grub"},
|
|
{name: "grub", gadgetFile: "grub.conf", opts: &bootloader.Options{Role: bootloader.RoleRunMode, NoSlashBoot: true}, expName: "grub"},
|
|
{name: "grub", gadgetFile: "grub.conf", opts: &bootloader.Options{Role: bootloader.RoleRecovery}, expName: "grub"},
|
|
{name: "uboot", gadgetFile: "uboot.conf", expName: "uboot"},
|
|
{name: "androidboot", gadgetFile: "androidboot.conf", expName: "androidboot"},
|
|
{name: "lk", gadgetFile: "lk.conf", expName: "lk"},
|
|
} {
|
|
c.Logf("tc: %v", tc.name)
|
|
gadgetDir := c.MkDir()
|
|
rootDir := c.MkDir()
|
|
err := os.MkdirAll(filepath.Join(rootDir, filepath.Dir(tc.gadgetFile)), 0755)
|
|
c.Assert(err, IsNil)
|
|
err = ioutil.WriteFile(filepath.Join(gadgetDir, tc.gadgetFile), nil, 0644)
|
|
c.Assert(err, IsNil)
|
|
bl, err := bootloader.ForGadget(gadgetDir, rootDir, tc.opts)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(bl, NotNil)
|
|
c.Check(bl.Name(), Equals, tc.expName)
|
|
}
|
|
}
|
|
|
|
func (s *bootenvTestSuite) TestBootFileWithPath(c *C) {
|
|
a := bootloader.NewBootFile("", "some/path", bootloader.RoleRunMode)
|
|
b := a.WithPath("other/path")
|
|
c.Assert(a.Path, Equals, "some/path")
|
|
c.Assert(b.Path, Equals, "other/path")
|
|
}
|