forked from apple/containerization
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathEXT4+FileTree.swift
More file actions
115 lines (107 loc) · 4.12 KB
/
EXT4+FileTree.swift
File metadata and controls
115 lines (107 loc) · 4.12 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
//===----------------------------------------------------------------------===//
// Copyright © 2025 Apple Inc. and the Containerization project authors. 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
//
// https://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.
//===----------------------------------------------------------------------===//
import Foundation
import SystemPackage
extension EXT4 {
class FileTree {
class FileTreeNode {
let inode: InodeNumber
let name: String
var children: [Ptr<FileTreeNode>] = []
var blocks: (start: UInt32, end: UInt32)?
var additionalBlocks: [(start: UInt32, end: UInt32)]?
var link: InodeNumber?
private var parent: Ptr<FileTreeNode>?
init(
inode: InodeNumber,
name: String,
parent: Ptr<FileTreeNode>?,
children: [Ptr<FileTreeNode>] = [],
blocks: (start: UInt32, end: UInt32)? = nil,
additionalBlocks: [(start: UInt32, end: UInt32)]? = nil,
link: InodeNumber? = nil
) {
self.inode = inode
self.name = name
self.children = children
self.blocks = blocks
self.additionalBlocks = additionalBlocks
self.link = link
self.parent = parent
}
deinit {
self.children.removeAll()
self.children = []
self.blocks = nil
self.additionalBlocks = nil
self.link = nil
}
var path: FilePath? {
var components: [String] = [self.name]
var _ptr = self.parent
while let ptr = _ptr {
components.append(ptr.pointee.name)
_ptr = ptr.pointee.parent
}
guard let last = components.last else {
return nil
}
guard components.count > 1 else {
return FilePath(last)
}
components = components.dropLast()
let path = components.reversed().joined(separator: "/")
guard let data = path.data(using: .utf8) else {
return nil
}
guard let dataPath = String(data: data, encoding: .utf8) else {
return nil
}
return FilePath(dataPath).pushing(FilePath(last)).lexicallyNormalized()
}
}
var root: Ptr<FileTreeNode>
init(_ root: InodeNumber, _ name: String) {
self.root = Ptr<FileTreeNode>.allocate(capacity: 1)
self.root.initialize(to: FileTreeNode(inode: root, name: name, parent: nil))
}
func lookup(path: FilePath) -> Ptr<FileTreeNode>? {
var components: [String] = path.items
var node = self.root
if components.first == "/" {
components = Array(components.dropFirst())
}
if components.count == 0 {
return node
}
for component in components {
var found = false
for childPtr in node.pointee.children {
let child = childPtr.pointee
if child.name == component {
node = childPtr
found = true
break
}
}
guard found else {
return nil
}
}
return node
}
}
}