Skip to content

Commit

Permalink
fix broken native and zran image support
Browse files Browse the repository at this point in the history
1. We should ignore to handle TOC when only bootstrap be provided;
2. Get zran rafs blob digest/size from CLI options for merge subcommand;

Signed-off-by: Yan Song <[email protected]>
  • Loading branch information
imeoer committed Dec 28, 2022
1 parent c4090c9 commit 27402e5
Show file tree
Hide file tree
Showing 9 changed files with 69 additions and 49 deletions.
16 changes: 13 additions & 3 deletions api/src/config.rs
Original file line number Diff line number Diff line change
Expand Up @@ -145,9 +145,12 @@ impl ConfigV2 {

/// Get configuration information for RAFS filesystem.
pub fn get_rafs_config(&self) -> Result<&RafsConfigV2> {
self.rafs
.as_ref()
.ok_or_else(|| einval!("no configuration information for rafs"))
self.rafs.as_ref().ok_or_else(|| {
Error::new(
ErrorKind::InvalidInput,
"no configuration information for rafs",
)
})
}

/// Clone the object with all secrets removed.
Expand Down Expand Up @@ -644,6 +647,9 @@ pub struct RafsConfigV2 {
/// Filesystem prefetching configuration.
#[serde(default)]
pub prefetch: PrefetchConfigV2,
/// Probe TOC list included in the blob.
#[serde(default)]
pub probe_toc: bool,
}

impl RafsConfigV2 {
Expand Down Expand Up @@ -969,6 +975,9 @@ struct RafsConfig {
// ZERO value means, amplifying user io is not enabled.
#[serde(default = "default_batch_size")]
pub amplify_io: usize,
/// Probe TOC list included in the blob.
#[serde(default)]
pub probe_toc: bool,
}

impl TryFrom<RafsConfig> for ConfigV2 {
Expand All @@ -986,6 +995,7 @@ impl TryFrom<RafsConfig> for ConfigV2 {
access_pattern: v.access_pattern,
latest_read_files: v.latest_read_files,
prefetch: v.fs_prefetch.into(),
probe_toc: v.probe_toc,
};
if !cache.prefetch.enable && rafs.prefetch.enable {
cache.prefetch = rafs.prefetch.clone();
Expand Down
5 changes: 4 additions & 1 deletion rafs/src/fs.rs
Original file line number Diff line number Diff line change
Expand Up @@ -86,7 +86,10 @@ impl Rafs {
let blob_infos = sb.superblock.get_blob_infos();
let device = BlobDevice::new(cfg, &blob_infos).map_err(RafsError::CreateDevice)?;

if cfg.is_chunk_validation_enabled() && sb.meta.has_inlined_chunk_digest() {
if rafs_cfg.probe_toc
&& cfg.is_chunk_validation_enabled()
&& sb.meta.has_inlined_chunk_digest()
{
sb.superblock.set_blob_device(device.clone());
}

Expand Down
2 changes: 1 addition & 1 deletion rafs/src/metadata/direct_v6.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1152,7 +1152,7 @@ impl RafsInodeExt for OndiskInodeWrapper {
+ (idx as usize * size_of::<RafsV6InodeChunkAddr>());
let chunk_addr = state.map.get_ref::<RafsV6InodeChunkAddr>(offset)?;

if state.meta.has_inlined_chunk_digest() {
if state.meta.probe_toc && state.meta.has_inlined_chunk_digest() {
let blob_index = chunk_addr.blob_index();
let chunk_index = chunk_addr.blob_ci_index();
let device = self.mapping.device.lock().unwrap();
Expand Down
52 changes: 34 additions & 18 deletions rafs/src/metadata/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -431,6 +431,9 @@ pub struct RafsSuperMeta {
pub chunk_table_offset: u64,
/// Size of the chunk table for RAFS v6.
pub chunk_table_size: u64,

/// Probe TOC list included in the blob.
pub probe_toc: bool,
}

impl RafsSuperMeta {
Expand Down Expand Up @@ -519,6 +522,7 @@ impl Default for RafsSuperMeta {
is_chunk_dict: false,
chunk_table_offset: 0,
chunk_table_size: 0,
probe_toc: false,
}
}
}
Expand Down Expand Up @@ -634,9 +638,12 @@ impl Default for RafsSuper {
impl RafsSuper {
/// Create a new `RafsSuper` instance from a `RafsConfigV2` object.
pub fn new(conf: &RafsConfigV2) -> Result<Self> {
let mut meta = RafsSuperMeta::default();
meta.probe_toc = conf.probe_toc;
Ok(Self {
mode: RafsMode::from_str(conf.mode.as_str())?,
validate_digest: conf.validate,
meta,
..Default::default()
})
}
Expand All @@ -655,12 +662,18 @@ impl RafsSuper {
is_chunk_dict: bool,
config: Arc<ConfigV2>,
) -> Result<(Self, RafsIoReader)> {
let mut probe_toc = false;
if let Ok(rafs_config) = config.get_rafs_config() {
probe_toc = rafs_config.probe_toc;
}

let mut rs = RafsSuper {
mode: RafsMode::Direct,
validate_digest,
..Default::default()
};
rs.meta.is_chunk_dict = is_chunk_dict;
rs.meta.probe_toc = probe_toc;

// open bootstrap file
let file = OpenOptions::new()
Expand All @@ -669,25 +682,28 @@ impl RafsSuper {
.open(path.as_ref())?;
let mut reader = Box::new(file) as RafsIoReader;

if let Err(e) = rs.load(&mut reader) {
let id = BlobInfo::get_blob_id_from_meta_path(path.as_ref())?;
let new_path = match TocEntryList::extract_rafs_meta(&id, config.clone()) {
Ok(v) => v,
Err(_e) => {
debug!("failed to load inlined RAFS meta, {}", _e);
return Err(e);
}
};
let file = OpenOptions::new().read(true).write(false).open(new_path)?;
reader = Box::new(file) as RafsIoReader;
if probe_toc {
if let Err(e) = rs.load(&mut reader) {
let id = BlobInfo::get_blob_id_from_meta_path(path.as_ref())?;
let new_path = match TocEntryList::extract_rafs_meta(&id, config.clone()) {
Ok(v) => v,
Err(_e) => {
debug!("failed to load inlined RAFS meta, {}", _e);
return Err(e);
}
};
let file = OpenOptions::new().read(true).write(false).open(new_path)?;
reader = Box::new(file) as RafsIoReader;
rs.load(&mut reader)?;
rs.set_blob_id_from_meta_path(path.as_ref())?;
}
if (validate_digest || config.is_chunk_validation_enabled())
&& rs.meta.has_inlined_chunk_digest()
{
rs.create_blob_device(config)?;
}
} else {
rs.load(&mut reader)?;
rs.set_blob_id_from_meta_path(path.as_ref())?;
}

if (validate_digest || config.is_chunk_validation_enabled())
&& rs.meta.has_inlined_chunk_digest()
{
rs.create_blob_device(config)?;
}

Ok((rs, reader))
Expand Down
12 changes: 2 additions & 10 deletions src/bin/nydus-image/builder/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -84,11 +84,7 @@ fn dump_bootstrap(
}
}
if !ctx.conversion_type.is_to_ref() {
if ctx.blob_inline_meta {
blob_ctx.compressed_blob_size = 0;
} else {
blob_ctx.compressed_blob_size = blob_writer.pos()?;
}
blob_ctx.compressed_blob_size = blob_writer.pos()?;
}
}

Expand Down Expand Up @@ -174,11 +170,7 @@ fn finalize_blob(
dump_toc(ctx, blob_ctx, blob_writer)?;

if !ctx.conversion_type.is_to_ref() {
if ctx.blob_inline_meta {
blob_ctx.compressed_blob_size = 0;
} else {
blob_ctx.compressed_blob_size = blob_writer.pos()?;
}
blob_ctx.compressed_blob_size = blob_writer.pos()?;
}
if ctx.blob_inline_meta && blob_ctx.blob_id == "x".repeat(64) {
blob_ctx.blob_id = String::new();
Expand Down
2 changes: 1 addition & 1 deletion src/bin/nydus-image/core/chunk_dict.rs
Original file line number Diff line number Diff line change
Expand Up @@ -133,7 +133,7 @@ impl HashChunkDict {
};

rafs_config.check_compatibility(&rs.meta)?;
if rs.meta.is_v5() || rs.meta.has_inlined_chunk_digest() {
if rs.meta.is_v5() || rs.meta.probe_toc && rs.meta.has_inlined_chunk_digest() {
Tree::from_bootstrap(&rs, &mut d).context("failed to build tree from bootstrap")?;
} else if rs.meta.is_v6() {
d.load_chunk_table(&rs)
Expand Down
8 changes: 7 additions & 1 deletion src/bin/nydus-image/core/context.rs
Original file line number Diff line number Diff line change
Expand Up @@ -472,8 +472,14 @@ impl BlobContext {
let mut blob_id = blob.raw_blob_id().to_string();
let mut features = blob.features();

let mut probe_toc = false;
if let Ok(rafs_config) = ctx.configuration.get_rafs_config() {
probe_toc = rafs_config.probe_toc;
}

// Fixes up blob info objects from inlined-meta blobs.
if features.contains(BlobFeatures::INLINED_META)
if probe_toc
&& features.contains(BlobFeatures::INLINED_META)
&& (chunk_source == ChunkSource::Dict || chunk_source == ChunkSource::Parent)
{
let backend_config = ctx
Expand Down
4 changes: 1 addition & 3 deletions src/bin/nydus-image/merge.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,6 @@ use crate::core::tree::{MetadataTreeBuilder, Tree};
pub struct Merger {}

impl Merger {
/*
/// Merge assumes the bootstrap name is in `$blob_digest` format.
fn get_raw_blob_digest(bootstrap_path: &Path) -> Result<String> {
let file_name = bootstrap_path
Expand All @@ -41,7 +40,6 @@ impl Merger {
.ok_or_else(|| anyhow!("convert to string"))?;
Ok(file_name.to_string())
}
*/

fn get_digest_from_list(digests: &Option<Vec<String>>, idx: usize) -> Result<Option<[u8; 32]>> {
Ok(if let Some(digests) = &digests {
Expand Down Expand Up @@ -163,7 +161,7 @@ impl Merger {

// The blob id (blob sha256 hash) in parent bootstrap is invalid for nydusd
// runtime, should change it to the hash of whole tar blob.
blob_ctx.blob_id = blob.blob_id().to_owned();
blob_ctx.blob_id = Self::get_raw_blob_digest(bootstrap_path)?;
if let Some(digest) = Self::get_digest_from_list(&blob_digests, layer_idx)? {
if blob.has_feature(BlobFeatures::ZRAN) {
blob_ctx.rafs_blob_digest = digest;
Expand Down
17 changes: 6 additions & 11 deletions storage/src/device.rs
Original file line number Diff line number Diff line change
Expand Up @@ -196,9 +196,7 @@ impl BlobInfo {

/// Get the id of the blob, with special handling of `inlined-meta` case.
pub fn blob_id(&self) -> String {
if self.has_feature(BlobFeatures::INLINED_META) && !self.has_feature(BlobFeatures::ZRAN)
|| !self.meta_ci_is_valid()
{
if self.has_feature(BlobFeatures::INLINED_META) && !self.has_feature(BlobFeatures::ZRAN) {
let guard = self.meta_path.lock().unwrap();
if !guard.is_empty() {
return guard.deref().clone();
Expand Down Expand Up @@ -451,16 +449,13 @@ impl BlobInfo {
/// Get RAFS blob id for ZRan.
pub fn get_rafs_blob_id(&self) -> Result<String, Error> {
assert!(self.has_feature(BlobFeatures::ZRAN));
let id = if self.has_feature(BlobFeatures::INLINED_META) {
if self.has_feature(BlobFeatures::INLINED_META) {
let guard = self.meta_path.lock().unwrap();
if guard.is_empty() {
return Err(einval!("failed to get blob id from meta file name"));
if !guard.is_empty() {
return Ok(guard.deref().clone());
}
guard.deref().clone()
} else {
hex::encode(&self.rafs_blob_digest)
};
Ok(id)
}
Ok(hex::encode(&self.rafs_blob_digest))
}
}

Expand Down

0 comments on commit 27402e5

Please sign in to comment.