forked from mirrors/jj
index: add a newtype wrapper for IndexPosition
It seems better to both hide the specific type and to get some more type safety.
This commit is contained in:
parent
5b18e89a4d
commit
0145be3693
2 changed files with 121 additions and 90 deletions
207
lib/src/index.rs
207
lib/src/index.rs
|
@ -32,6 +32,9 @@ use tempfile::NamedTempFile;
|
||||||
use crate::commit::Commit;
|
use crate::commit::Commit;
|
||||||
use crate::store::{ChangeId, CommitId};
|
use crate::store::{ChangeId, CommitId};
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Hash)]
|
||||||
|
pub struct IndexPosition(u32);
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub enum IndexRef<'a> {
|
pub enum IndexRef<'a> {
|
||||||
Readonly(&'a ReadonlyIndex),
|
Readonly(&'a ReadonlyIndex),
|
||||||
|
@ -65,7 +68,7 @@ impl<'a> IndexRef<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<u32> {
|
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||||
match self {
|
match self {
|
||||||
IndexRef::Readonly(index) => index.commit_id_to_pos(commit_id),
|
IndexRef::Readonly(index) => index.commit_id_to_pos(commit_id),
|
||||||
IndexRef::Mutable(index) => index.commit_id_to_pos(commit_id),
|
IndexRef::Mutable(index) => index.commit_id_to_pos(commit_id),
|
||||||
|
@ -86,7 +89,7 @@ impl<'a> IndexRef<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn entry_by_pos(&self, pos: u32) -> IndexEntry<'a> {
|
pub fn entry_by_pos(&self, pos: IndexPosition) -> IndexEntry<'a> {
|
||||||
match self {
|
match self {
|
||||||
IndexRef::Readonly(index) => index.entry_by_pos(pos),
|
IndexRef::Readonly(index) => index.entry_by_pos(pos),
|
||||||
IndexRef::Mutable(index) => index.entry_by_pos(pos),
|
IndexRef::Mutable(index) => index.entry_by_pos(pos),
|
||||||
|
@ -156,8 +159,8 @@ impl CommitGraphEntry<'_> {
|
||||||
(&self.data[8..]).read_u32::<LittleEndian>().unwrap()
|
(&self.data[8..]).read_u32::<LittleEndian>().unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parent1_pos(&self) -> u32 {
|
fn parent1_pos(&self) -> IndexPosition {
|
||||||
(&self.data[12..]).read_u32::<LittleEndian>().unwrap()
|
IndexPosition((&self.data[12..]).read_u32::<LittleEndian>().unwrap())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parent2_overflow_pos(&self) -> u32 {
|
fn parent2_overflow_pos(&self) -> u32 {
|
||||||
|
@ -168,8 +171,8 @@ impl CommitGraphEntry<'_> {
|
||||||
(&self.data[20..]).read_u32::<LittleEndian>().unwrap()
|
(&self.data[20..]).read_u32::<LittleEndian>().unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn predecessor1_pos(&self) -> u32 {
|
fn predecessor1_pos(&self) -> IndexPosition {
|
||||||
(&self.data[24..]).read_u32::<LittleEndian>().unwrap()
|
IndexPosition((&self.data[24..]).read_u32::<LittleEndian>().unwrap())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn predecessor2_overflow_pos(&self) -> u32 {
|
fn predecessor2_overflow_pos(&self) -> u32 {
|
||||||
|
@ -205,10 +208,12 @@ impl CommitLookupEntry<'_> {
|
||||||
CommitId(self.data[0..self.hash_length].to_vec())
|
CommitId(self.data[0..self.hash_length].to_vec())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn pos(&self) -> u32 {
|
fn pos(&self) -> IndexPosition {
|
||||||
(&self.data[self.hash_length..self.hash_length + 4])
|
IndexPosition(
|
||||||
.read_u32::<LittleEndian>()
|
(&self.data[self.hash_length..self.hash_length + 4])
|
||||||
.unwrap()
|
.read_u32::<LittleEndian>()
|
||||||
|
.unwrap(),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -315,8 +320,8 @@ struct MutableGraphEntry {
|
||||||
change_id: ChangeId,
|
change_id: ChangeId,
|
||||||
is_pruned: bool,
|
is_pruned: bool,
|
||||||
generation_number: u32,
|
generation_number: u32,
|
||||||
parent_positions: Vec<u32>,
|
parent_positions: Vec<IndexPosition>,
|
||||||
predecessor_positions: Vec<u32>,
|
predecessor_positions: Vec<IndexPosition>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct MutableIndex {
|
pub struct MutableIndex {
|
||||||
|
@ -324,7 +329,7 @@ pub struct MutableIndex {
|
||||||
num_parent_commits: u32,
|
num_parent_commits: u32,
|
||||||
hash_length: usize,
|
hash_length: usize,
|
||||||
graph: Vec<MutableGraphEntry>,
|
graph: Vec<MutableGraphEntry>,
|
||||||
lookup: BTreeMap<CommitId, u32>,
|
lookup: BTreeMap<CommitId, IndexPosition>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MutableIndex {
|
impl MutableIndex {
|
||||||
|
@ -401,7 +406,7 @@ impl MutableIndex {
|
||||||
}
|
}
|
||||||
self.lookup.insert(
|
self.lookup.insert(
|
||||||
entry.commit_id.clone(),
|
entry.commit_id.clone(),
|
||||||
self.graph.len() as u32 + self.num_parent_commits,
|
IndexPosition(self.graph.len() as u32 + self.num_parent_commits),
|
||||||
);
|
);
|
||||||
self.graph.push(entry);
|
self.graph.push(entry);
|
||||||
}
|
}
|
||||||
|
@ -409,7 +414,7 @@ impl MutableIndex {
|
||||||
fn add_commits_from(&mut self, other_segment: &dyn IndexSegment) {
|
fn add_commits_from(&mut self, other_segment: &dyn IndexSegment) {
|
||||||
let other = CompositeIndex(other_segment);
|
let other = CompositeIndex(other_segment);
|
||||||
for pos in other_segment.segment_num_parent_commits()..other.num_commits() {
|
for pos in other_segment.segment_num_parent_commits()..other.num_commits() {
|
||||||
let entry = other.entry_by_pos(pos);
|
let entry = other.entry_by_pos(IndexPosition(pos));
|
||||||
let parent_ids: Vec<_> = entry
|
let parent_ids: Vec<_> = entry
|
||||||
.parents()
|
.parents()
|
||||||
.iter()
|
.iter()
|
||||||
|
@ -495,7 +500,7 @@ impl MutableIndex {
|
||||||
|
|
||||||
buf.write_u32::<LittleEndian>(entry.parent_positions.len() as u32)
|
buf.write_u32::<LittleEndian>(entry.parent_positions.len() as u32)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let mut parent1_pos = 0;
|
let mut parent1_pos = IndexPosition(0);
|
||||||
let parent_overflow_pos = parent_overflow.len() as u32;
|
let parent_overflow_pos = parent_overflow.len() as u32;
|
||||||
for (i, parent_pos) in entry.parent_positions.iter().enumerate() {
|
for (i, parent_pos) in entry.parent_positions.iter().enumerate() {
|
||||||
if i == 0 {
|
if i == 0 {
|
||||||
|
@ -504,12 +509,12 @@ impl MutableIndex {
|
||||||
parent_overflow.push(*parent_pos);
|
parent_overflow.push(*parent_pos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
buf.write_u32::<LittleEndian>(parent1_pos).unwrap();
|
buf.write_u32::<LittleEndian>(parent1_pos.0).unwrap();
|
||||||
buf.write_u32::<LittleEndian>(parent_overflow_pos).unwrap();
|
buf.write_u32::<LittleEndian>(parent_overflow_pos).unwrap();
|
||||||
|
|
||||||
buf.write_u32::<LittleEndian>(entry.predecessor_positions.len() as u32)
|
buf.write_u32::<LittleEndian>(entry.predecessor_positions.len() as u32)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let mut predecessor1_pos = 0;
|
let mut predecessor1_pos = IndexPosition(0);
|
||||||
let predecessor_overflow_pos = predecessor_overflow.len() as u32;
|
let predecessor_overflow_pos = predecessor_overflow.len() as u32;
|
||||||
for (i, predecessor_pos) in entry.predecessor_positions.iter().enumerate() {
|
for (i, predecessor_pos) in entry.predecessor_positions.iter().enumerate() {
|
||||||
if i == 0 {
|
if i == 0 {
|
||||||
|
@ -518,7 +523,7 @@ impl MutableIndex {
|
||||||
predecessor_overflow.push(*predecessor_pos);
|
predecessor_overflow.push(*predecessor_pos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
buf.write_u32::<LittleEndian>(predecessor1_pos).unwrap();
|
buf.write_u32::<LittleEndian>(predecessor1_pos.0).unwrap();
|
||||||
buf.write_u32::<LittleEndian>(predecessor_overflow_pos)
|
buf.write_u32::<LittleEndian>(predecessor_overflow_pos)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
@ -531,7 +536,7 @@ impl MutableIndex {
|
||||||
|
|
||||||
for (commit_id, pos) in self.lookup {
|
for (commit_id, pos) in self.lookup {
|
||||||
buf.write_all(commit_id.0.as_slice()).unwrap();
|
buf.write_all(commit_id.0.as_slice()).unwrap();
|
||||||
buf.write_u32::<LittleEndian>(pos).unwrap();
|
buf.write_u32::<LittleEndian>(pos.0).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
buf[parent_overflow_offset..parent_overflow_offset + 4]
|
buf[parent_overflow_offset..parent_overflow_offset + 4]
|
||||||
|
@ -539,14 +544,14 @@ impl MutableIndex {
|
||||||
.write_u32::<LittleEndian>(parent_overflow.len() as u32)
|
.write_u32::<LittleEndian>(parent_overflow.len() as u32)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
for parent_pos in parent_overflow {
|
for parent_pos in parent_overflow {
|
||||||
buf.write_u32::<LittleEndian>(parent_pos).unwrap();
|
buf.write_u32::<LittleEndian>(parent_pos.0).unwrap();
|
||||||
}
|
}
|
||||||
buf[predecessor_overflow_offset..predecessor_overflow_offset + 4]
|
buf[predecessor_overflow_offset..predecessor_overflow_offset + 4]
|
||||||
.as_mut()
|
.as_mut()
|
||||||
.write_u32::<LittleEndian>(predecessor_overflow.len() as u32)
|
.write_u32::<LittleEndian>(predecessor_overflow.len() as u32)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
for predecessor_pos in predecessor_overflow {
|
for predecessor_pos in predecessor_overflow {
|
||||||
buf.write_u32::<LittleEndian>(predecessor_pos).unwrap();
|
buf.write_u32::<LittleEndian>(predecessor_pos.0).unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
buf
|
buf
|
||||||
|
@ -621,7 +626,7 @@ impl MutableIndex {
|
||||||
CompositeIndex(self).stats()
|
CompositeIndex(self).stats()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<u32> {
|
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||||
CompositeIndex(self).commit_id_to_pos(commit_id)
|
CompositeIndex(self).commit_id_to_pos(commit_id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -633,7 +638,7 @@ impl MutableIndex {
|
||||||
CompositeIndex(self).entry_by_id(commit_id)
|
CompositeIndex(self).entry_by_id(commit_id)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn entry_by_pos(&self, pos: u32) -> IndexEntry {
|
pub fn entry_by_pos(&self, pos: IndexPosition) -> IndexEntry {
|
||||||
CompositeIndex(self).entry_by_pos(pos)
|
CompositeIndex(self).entry_by_pos(pos)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -677,7 +682,7 @@ trait IndexSegment {
|
||||||
|
|
||||||
fn segment_name(&self) -> Option<String>;
|
fn segment_name(&self) -> Option<String>;
|
||||||
|
|
||||||
fn segment_commit_id_to_pos(&self, commit_id: &CommitId) -> Option<u32>;
|
fn segment_commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition>;
|
||||||
|
|
||||||
fn segment_resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution;
|
fn segment_resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution;
|
||||||
|
|
||||||
|
@ -691,13 +696,13 @@ trait IndexSegment {
|
||||||
|
|
||||||
fn segment_num_parents(&self, local_pos: u32) -> u32;
|
fn segment_num_parents(&self, local_pos: u32) -> u32;
|
||||||
|
|
||||||
fn segment_parent_positions(&self, local_pos: u32) -> Vec<u32>;
|
fn segment_parent_positions(&self, local_pos: u32) -> Vec<IndexPosition>;
|
||||||
|
|
||||||
fn segment_num_predecessors(&self, local_pos: u32) -> u32;
|
fn segment_num_predecessors(&self, local_pos: u32) -> u32;
|
||||||
|
|
||||||
fn segment_predecessor_positions(&self, local_pos: u32) -> Vec<u32>;
|
fn segment_predecessor_positions(&self, local_pos: u32) -> Vec<IndexPosition>;
|
||||||
|
|
||||||
fn segment_entry_by_pos(&self, pos: u32, local_pos: u32) -> IndexEntry;
|
fn segment_entry_by_pos(&self, pos: IndexPosition, local_pos: u32) -> IndexEntry;
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
|
@ -716,13 +721,13 @@ impl<'a> CompositeIndex<'a> {
|
||||||
let mut change_ids = HashSet::new();
|
let mut change_ids = HashSet::new();
|
||||||
let mut num_pruned_commits = 0;
|
let mut num_pruned_commits = 0;
|
||||||
for pos in 0..num_commits {
|
for pos in 0..num_commits {
|
||||||
let entry = self.entry_by_pos(pos);
|
let entry = self.entry_by_pos(IndexPosition(pos));
|
||||||
max_generation_number = max(max_generation_number, entry.generation_number());
|
max_generation_number = max(max_generation_number, entry.generation_number());
|
||||||
if entry.num_parents() > 1 {
|
if entry.num_parents() > 1 {
|
||||||
num_merges += 1;
|
num_merges += 1;
|
||||||
}
|
}
|
||||||
for parent_pos in entry.parent_positions() {
|
for parent_pos in entry.parent_positions() {
|
||||||
is_head[parent_pos as usize] = false;
|
is_head[parent_pos.0 as usize] = false;
|
||||||
}
|
}
|
||||||
if entry.is_pruned() {
|
if entry.is_pruned() {
|
||||||
num_pruned_commits += 1;
|
num_pruned_commits += 1;
|
||||||
|
@ -757,10 +762,10 @@ impl<'a> CompositeIndex<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn entry_by_pos(&self, pos: u32) -> IndexEntry<'a> {
|
fn entry_by_pos(&self, pos: IndexPosition) -> IndexEntry<'a> {
|
||||||
let num_parent_commits = self.0.segment_num_parent_commits();
|
let num_parent_commits = self.0.segment_num_parent_commits();
|
||||||
if pos >= num_parent_commits {
|
if pos.0 >= num_parent_commits {
|
||||||
self.0.segment_entry_by_pos(pos, pos - num_parent_commits)
|
self.0.segment_entry_by_pos(pos, pos.0 - num_parent_commits)
|
||||||
} else {
|
} else {
|
||||||
let parent_file: &ReadonlyIndex =
|
let parent_file: &ReadonlyIndex =
|
||||||
self.0.segment_parent_file().as_ref().unwrap().as_ref();
|
self.0.segment_parent_file().as_ref().unwrap().as_ref();
|
||||||
|
@ -771,7 +776,7 @@ impl<'a> CompositeIndex<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<u32> {
|
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||||
let local_match = self.0.segment_commit_id_to_pos(commit_id);
|
let local_match = self.0.segment_commit_id_to_pos(commit_id);
|
||||||
local_match.or_else(|| {
|
local_match.or_else(|| {
|
||||||
self.0
|
self.0
|
||||||
|
@ -812,7 +817,7 @@ impl<'a> CompositeIndex<'a> {
|
||||||
self.is_ancestor_pos(ancestor_pos, descendant_pos)
|
self.is_ancestor_pos(ancestor_pos, descendant_pos)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn is_ancestor_pos(&self, ancestor_pos: u32, descendant_pos: u32) -> bool {
|
fn is_ancestor_pos(&self, ancestor_pos: IndexPosition, descendant_pos: IndexPosition) -> bool {
|
||||||
let ancestor_generation = self.entry_by_pos(ancestor_pos).generation_number();
|
let ancestor_generation = self.entry_by_pos(ancestor_pos).generation_number();
|
||||||
let mut work = vec![descendant_pos];
|
let mut work = vec![descendant_pos];
|
||||||
let mut visited = HashSet::new();
|
let mut visited = HashSet::new();
|
||||||
|
@ -848,7 +853,11 @@ impl<'a> CompositeIndex<'a> {
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn common_ancestors_pos(&self, set1: &[u32], set2: &[u32]) -> BTreeSet<u32> {
|
fn common_ancestors_pos(
|
||||||
|
&self,
|
||||||
|
set1: &[IndexPosition],
|
||||||
|
set2: &[IndexPosition],
|
||||||
|
) -> BTreeSet<IndexPosition> {
|
||||||
let mut items1: BTreeSet<_> = set1
|
let mut items1: BTreeSet<_> = set1
|
||||||
.iter()
|
.iter()
|
||||||
.map(|pos| IndexEntryByGeneration(self.entry_by_pos(*pos)))
|
.map(|pos| IndexEntryByGeneration(self.entry_by_pos(*pos)))
|
||||||
|
@ -911,7 +920,10 @@ impl<'a> CompositeIndex<'a> {
|
||||||
.collect()
|
.collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn heads_pos(&self, mut candidate_positions: BTreeSet<u32>) -> BTreeSet<u32> {
|
fn heads_pos(
|
||||||
|
&self,
|
||||||
|
mut candidate_positions: BTreeSet<IndexPosition>,
|
||||||
|
) -> BTreeSet<IndexPosition> {
|
||||||
// Add all parents of the candidates to the work queue. The parents and their
|
// Add all parents of the candidates to the work queue. The parents and their
|
||||||
// ancestors are not heads.
|
// ancestors are not heads.
|
||||||
// Also find the smallest generation number among the candidates.
|
// Also find the smallest generation number among the candidates.
|
||||||
|
@ -1017,8 +1029,8 @@ struct RevWalkWorkItem<'a> {
|
||||||
pub struct RevWalk<'a> {
|
pub struct RevWalk<'a> {
|
||||||
index: CompositeIndex<'a>,
|
index: CompositeIndex<'a>,
|
||||||
items: BinaryHeap<RevWalkWorkItem<'a>>,
|
items: BinaryHeap<RevWalkWorkItem<'a>>,
|
||||||
wanted_boundary_set: HashSet<u32>,
|
wanted_boundary_set: HashSet<IndexPosition>,
|
||||||
unwanted_boundary_set: HashSet<u32>,
|
unwanted_boundary_set: HashSet<IndexPosition>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> RevWalk<'a> {
|
impl<'a> RevWalk<'a> {
|
||||||
|
@ -1031,7 +1043,7 @@ impl<'a> RevWalk<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn add_wanted(&mut self, pos: u32) {
|
fn add_wanted(&mut self, pos: IndexPosition) {
|
||||||
if !self.wanted_boundary_set.insert(pos) {
|
if !self.wanted_boundary_set.insert(pos) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -1041,7 +1053,7 @@ impl<'a> RevWalk<'a> {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
fn add_unwanted(&mut self, pos: u32) {
|
fn add_unwanted(&mut self, pos: IndexPosition) {
|
||||||
if !self.unwanted_boundary_set.insert(pos) {
|
if !self.unwanted_boundary_set.insert(pos) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -1095,7 +1107,7 @@ impl IndexSegment for ReadonlyIndex {
|
||||||
Some(self.name.clone())
|
Some(self.name.clone())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn segment_commit_id_to_pos(&self, commit_id: &CommitId) -> Option<u32> {
|
fn segment_commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||||
if self.num_local_commits == 0 {
|
if self.num_local_commits == 0 {
|
||||||
// Avoid overflow when subtracting 1 below
|
// Avoid overflow when subtracting 1 below
|
||||||
return None;
|
return None;
|
||||||
|
@ -1129,7 +1141,7 @@ impl IndexSegment for ReadonlyIndex {
|
||||||
None => PrefixResolution::NoMatch,
|
None => PrefixResolution::NoMatch,
|
||||||
Some(lookup_pos) => {
|
Some(lookup_pos) => {
|
||||||
let mut first_match = None;
|
let mut first_match = None;
|
||||||
for i in lookup_pos..self.num_local_commits as u32 {
|
for i in lookup_pos.0..self.num_local_commits as u32 {
|
||||||
let entry = self.lookup_entry(i);
|
let entry = self.lookup_entry(i);
|
||||||
let id = entry.commit_id();
|
let id = entry.commit_id();
|
||||||
if !id.0.starts_with(&bytes_prefix.0) {
|
if !id.0.starts_with(&bytes_prefix.0) {
|
||||||
|
@ -1171,7 +1183,7 @@ impl IndexSegment for ReadonlyIndex {
|
||||||
self.graph_entry(local_pos).num_parents()
|
self.graph_entry(local_pos).num_parents()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn segment_parent_positions(&self, local_pos: u32) -> Vec<u32> {
|
fn segment_parent_positions(&self, local_pos: u32) -> Vec<IndexPosition> {
|
||||||
let graph_entry = self.graph_entry(local_pos);
|
let graph_entry = self.graph_entry(local_pos);
|
||||||
let mut parent_entries = vec![];
|
let mut parent_entries = vec![];
|
||||||
if graph_entry.num_parents() >= 1 {
|
if graph_entry.num_parents() >= 1 {
|
||||||
|
@ -1191,7 +1203,7 @@ impl IndexSegment for ReadonlyIndex {
|
||||||
self.graph_entry(local_pos).num_predecessors()
|
self.graph_entry(local_pos).num_predecessors()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn segment_predecessor_positions(&self, local_pos: u32) -> Vec<u32> {
|
fn segment_predecessor_positions(&self, local_pos: u32) -> Vec<IndexPosition> {
|
||||||
let graph_entry = self.graph_entry(local_pos);
|
let graph_entry = self.graph_entry(local_pos);
|
||||||
let mut predecessor_entries = vec![];
|
let mut predecessor_entries = vec![];
|
||||||
if graph_entry.num_predecessors() >= 1 {
|
if graph_entry.num_predecessors() >= 1 {
|
||||||
|
@ -1207,7 +1219,7 @@ impl IndexSegment for ReadonlyIndex {
|
||||||
predecessor_entries
|
predecessor_entries
|
||||||
}
|
}
|
||||||
|
|
||||||
fn segment_entry_by_pos(&self, pos: u32, local_pos: u32) -> IndexEntry {
|
fn segment_entry_by_pos(&self, pos: IndexPosition, local_pos: u32) -> IndexEntry {
|
||||||
IndexEntry {
|
IndexEntry {
|
||||||
source: self,
|
source: self,
|
||||||
local_pos,
|
local_pos,
|
||||||
|
@ -1233,7 +1245,7 @@ impl IndexSegment for MutableIndex {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|
||||||
fn segment_commit_id_to_pos(&self, commit_id: &CommitId) -> Option<u32> {
|
fn segment_commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||||
self.lookup.get(commit_id).cloned()
|
self.lookup.get(commit_id).cloned()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1287,7 +1299,7 @@ impl IndexSegment for MutableIndex {
|
||||||
self.graph[local_pos as usize].parent_positions.len() as u32
|
self.graph[local_pos as usize].parent_positions.len() as u32
|
||||||
}
|
}
|
||||||
|
|
||||||
fn segment_parent_positions(&self, local_pos: u32) -> Vec<u32> {
|
fn segment_parent_positions(&self, local_pos: u32) -> Vec<IndexPosition> {
|
||||||
self.graph[local_pos as usize].parent_positions.clone()
|
self.graph[local_pos as usize].parent_positions.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1295,11 +1307,11 @@ impl IndexSegment for MutableIndex {
|
||||||
self.graph[local_pos as usize].predecessor_positions.len() as u32
|
self.graph[local_pos as usize].predecessor_positions.len() as u32
|
||||||
}
|
}
|
||||||
|
|
||||||
fn segment_predecessor_positions(&self, local_pos: u32) -> Vec<u32> {
|
fn segment_predecessor_positions(&self, local_pos: u32) -> Vec<IndexPosition> {
|
||||||
self.graph[local_pos as usize].predecessor_positions.clone()
|
self.graph[local_pos as usize].predecessor_positions.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn segment_entry_by_pos(&self, pos: u32, local_pos: u32) -> IndexEntry {
|
fn segment_entry_by_pos(&self, pos: IndexPosition, local_pos: u32) -> IndexEntry {
|
||||||
IndexEntry {
|
IndexEntry {
|
||||||
source: self,
|
source: self,
|
||||||
local_pos,
|
local_pos,
|
||||||
|
@ -1311,7 +1323,7 @@ impl IndexSegment for MutableIndex {
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub struct IndexEntry<'a> {
|
pub struct IndexEntry<'a> {
|
||||||
source: &'a dyn IndexSegment,
|
source: &'a dyn IndexSegment,
|
||||||
pos: u32,
|
pos: IndexPosition,
|
||||||
// Position within the source segment
|
// Position within the source segment
|
||||||
local_pos: u32,
|
local_pos: u32,
|
||||||
}
|
}
|
||||||
|
@ -1339,7 +1351,7 @@ impl Hash for IndexEntry<'_> {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> IndexEntry<'a> {
|
impl<'a> IndexEntry<'a> {
|
||||||
pub fn position(&self) -> u32 {
|
pub fn position(&self) -> IndexPosition {
|
||||||
self.pos
|
self.pos
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1363,7 +1375,7 @@ impl<'a> IndexEntry<'a> {
|
||||||
self.source.segment_num_parents(self.local_pos)
|
self.source.segment_num_parents(self.local_pos)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn parent_positions(&self) -> Vec<u32> {
|
pub fn parent_positions(&self) -> Vec<IndexPosition> {
|
||||||
self.source.segment_parent_positions(self.local_pos)
|
self.source.segment_parent_positions(self.local_pos)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1379,7 +1391,7 @@ impl<'a> IndexEntry<'a> {
|
||||||
self.source.segment_num_predecessors(self.local_pos)
|
self.source.segment_num_predecessors(self.local_pos)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn predecessor_positions(&self) -> Vec<u32> {
|
pub fn predecessor_positions(&self) -> Vec<IndexPosition> {
|
||||||
self.source.segment_predecessor_positions(self.local_pos)
|
self.source.segment_predecessor_positions(self.local_pos)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1461,7 +1473,7 @@ impl ReadonlyIndex {
|
||||||
CompositeIndex(self).stats()
|
CompositeIndex(self).stats()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<u32> {
|
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||||
CompositeIndex(self).commit_id_to_pos(commit_id)
|
CompositeIndex(self).commit_id_to_pos(commit_id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1473,7 +1485,7 @@ impl ReadonlyIndex {
|
||||||
CompositeIndex(self).entry_by_id(commit_id)
|
CompositeIndex(self).entry_by_id(commit_id)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn entry_by_pos(&self, pos: u32) -> IndexEntry {
|
pub fn entry_by_pos(&self, pos: IndexPosition) -> IndexEntry {
|
||||||
CompositeIndex(self).entry_by_pos(pos)
|
CompositeIndex(self).entry_by_pos(pos)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1523,21 +1535,25 @@ impl ReadonlyIndex {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn overflow_parent(&self, overflow_pos: u32) -> u32 {
|
fn overflow_parent(&self, overflow_pos: u32) -> IndexPosition {
|
||||||
let offset = (overflow_pos as usize) * 4;
|
let offset = (overflow_pos as usize) * 4;
|
||||||
(&self.overflow_parent[offset..offset + 4])
|
IndexPosition(
|
||||||
.read_u32::<LittleEndian>()
|
(&self.overflow_parent[offset..offset + 4])
|
||||||
.unwrap()
|
.read_u32::<LittleEndian>()
|
||||||
|
.unwrap(),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn overflow_predecessor(&self, overflow_pos: u32) -> u32 {
|
fn overflow_predecessor(&self, overflow_pos: u32) -> IndexPosition {
|
||||||
let offset = (overflow_pos as usize) * 4;
|
let offset = (overflow_pos as usize) * 4;
|
||||||
(&self.overflow_predecessor[offset..offset + 4])
|
IndexPosition(
|
||||||
.read_u32::<LittleEndian>()
|
(&self.overflow_predecessor[offset..offset + 4])
|
||||||
.unwrap()
|
.read_u32::<LittleEndian>()
|
||||||
|
.unwrap(),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn commit_id_byte_prefix_to_pos(&self, prefix: &CommitId) -> Option<u32> {
|
fn commit_id_byte_prefix_to_pos(&self, prefix: &CommitId) -> Option<IndexPosition> {
|
||||||
if self.num_local_commits == 0 {
|
if self.num_local_commits == 0 {
|
||||||
// Avoid overflow when subtracting 1 below
|
// Avoid overflow when subtracting 1 below
|
||||||
return None;
|
return None;
|
||||||
|
@ -1553,7 +1569,7 @@ impl ReadonlyIndex {
|
||||||
let entry_commit_id = entry.commit_id();
|
let entry_commit_id = entry.commit_id();
|
||||||
let entry_prefix = &entry_commit_id.0[0..prefix_len];
|
let entry_prefix = &entry_commit_id.0[0..prefix_len];
|
||||||
if high == low {
|
if high == low {
|
||||||
return Some(mid);
|
return Some(IndexPosition(mid));
|
||||||
}
|
}
|
||||||
if entry_prefix < prefix.0.as_slice() {
|
if entry_prefix < prefix.0.as_slice() {
|
||||||
low = mid + 1;
|
low = mid + 1;
|
||||||
|
@ -1625,21 +1641,21 @@ mod tests {
|
||||||
assert_eq!(stats.num_pruned_commits, 0);
|
assert_eq!(stats.num_pruned_commits, 0);
|
||||||
assert_eq!(index.num_commits(), 1);
|
assert_eq!(index.num_commits(), 1);
|
||||||
// Can find only the root commit
|
// Can find only the root commit
|
||||||
assert_eq!(index.commit_id_to_pos(&id_0), Some(0));
|
assert_eq!(index.commit_id_to_pos(&id_0), Some(IndexPosition(0)));
|
||||||
assert_eq!(index.commit_id_to_pos(&CommitId::from_hex("aaaaaa")), None);
|
assert_eq!(index.commit_id_to_pos(&CommitId::from_hex("aaaaaa")), None);
|
||||||
assert_eq!(index.commit_id_to_pos(&CommitId::from_hex("ffffff")), None);
|
assert_eq!(index.commit_id_to_pos(&CommitId::from_hex("ffffff")), None);
|
||||||
// Check properties of root entry
|
// Check properties of root entry
|
||||||
let entry = index.entry_by_id(&id_0).unwrap();
|
let entry = index.entry_by_id(&id_0).unwrap();
|
||||||
assert_eq!(entry.pos, 0);
|
assert_eq!(entry.pos, IndexPosition(0));
|
||||||
assert_eq!(entry.commit_id(), id_0);
|
assert_eq!(entry.commit_id(), id_0);
|
||||||
assert_eq!(entry.change_id(), change_id0);
|
assert_eq!(entry.change_id(), change_id0);
|
||||||
assert!(!entry.is_pruned());
|
assert!(!entry.is_pruned());
|
||||||
assert_eq!(entry.generation_number(), 0);
|
assert_eq!(entry.generation_number(), 0);
|
||||||
assert_eq!(entry.num_parents(), 0);
|
assert_eq!(entry.num_parents(), 0);
|
||||||
assert_eq!(entry.parent_positions(), Vec::<u32>::new());
|
assert_eq!(entry.parent_positions(), Vec::<IndexPosition>::new());
|
||||||
assert_eq!(entry.parents(), Vec::<IndexEntry>::new());
|
assert_eq!(entry.parents(), Vec::<IndexEntry>::new());
|
||||||
assert_eq!(entry.num_predecessors(), 0);
|
assert_eq!(entry.num_predecessors(), 0);
|
||||||
assert_eq!(entry.predecessor_positions(), Vec::<u32>::new());
|
assert_eq!(entry.predecessor_positions(), Vec::<IndexPosition>::new());
|
||||||
assert_eq!(entry.predecessors(), Vec::<IndexEntry>::new());
|
assert_eq!(entry.predecessors(), Vec::<IndexEntry>::new());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1756,47 +1772,53 @@ mod tests {
|
||||||
let entry_4 = index.entry_by_id(&id_4).unwrap();
|
let entry_4 = index.entry_by_id(&id_4).unwrap();
|
||||||
let entry_5 = index.entry_by_id(&id_5).unwrap();
|
let entry_5 = index.entry_by_id(&id_5).unwrap();
|
||||||
// Check properties of some entries
|
// Check properties of some entries
|
||||||
assert_eq!(entry_0.pos, 0);
|
assert_eq!(entry_0.pos, IndexPosition(0));
|
||||||
assert_eq!(entry_0.commit_id(), id_0);
|
assert_eq!(entry_0.commit_id(), id_0);
|
||||||
assert_eq!(entry_1.pos, 1);
|
assert_eq!(entry_1.pos, IndexPosition(1));
|
||||||
assert_eq!(entry_1.commit_id(), id_1);
|
assert_eq!(entry_1.commit_id(), id_1);
|
||||||
assert_eq!(entry_1.change_id(), change_id1);
|
assert_eq!(entry_1.change_id(), change_id1);
|
||||||
assert!(!entry_1.is_pruned());
|
assert!(!entry_1.is_pruned());
|
||||||
assert_eq!(entry_1.generation_number(), 1);
|
assert_eq!(entry_1.generation_number(), 1);
|
||||||
assert_eq!(entry_1.num_parents(), 1);
|
assert_eq!(entry_1.num_parents(), 1);
|
||||||
assert_eq!(entry_1.parent_positions(), vec![0]);
|
assert_eq!(entry_1.parent_positions(), vec![IndexPosition(0)]);
|
||||||
assert_eq!(entry_1.parents().len(), 1);
|
assert_eq!(entry_1.parents().len(), 1);
|
||||||
assert_eq!(entry_1.parents()[0].pos, 0);
|
assert_eq!(entry_1.parents()[0].pos, IndexPosition(0));
|
||||||
assert_eq!(entry_1.num_predecessors(), 0);
|
assert_eq!(entry_1.num_predecessors(), 0);
|
||||||
assert_eq!(entry_1.predecessor_positions(), Vec::<u32>::new());
|
assert_eq!(entry_1.predecessor_positions(), Vec::<IndexPosition>::new());
|
||||||
assert_eq!(entry_2.pos, 2);
|
assert_eq!(entry_2.pos, IndexPosition(2));
|
||||||
assert_eq!(entry_2.commit_id(), id_2);
|
assert_eq!(entry_2.commit_id(), id_2);
|
||||||
assert_eq!(entry_2.change_id(), change_id2);
|
assert_eq!(entry_2.change_id(), change_id2);
|
||||||
assert!(!entry_2.is_pruned());
|
assert!(!entry_2.is_pruned());
|
||||||
assert_eq!(entry_2.generation_number(), 1);
|
assert_eq!(entry_2.generation_number(), 1);
|
||||||
assert_eq!(entry_2.num_parents(), 1);
|
assert_eq!(entry_2.num_parents(), 1);
|
||||||
assert_eq!(entry_2.parent_positions(), vec![0]);
|
assert_eq!(entry_2.parent_positions(), vec![IndexPosition(0)]);
|
||||||
assert_eq!(entry_3.change_id(), change_id3);
|
assert_eq!(entry_3.change_id(), change_id3);
|
||||||
assert_eq!(entry_3.generation_number(), 2);
|
assert_eq!(entry_3.generation_number(), 2);
|
||||||
assert_eq!(entry_3.parent_positions(), vec![2]);
|
assert_eq!(entry_3.parent_positions(), vec![IndexPosition(2)]);
|
||||||
assert!(entry_3.is_pruned());
|
assert!(entry_3.is_pruned());
|
||||||
assert_eq!(entry_4.pos, 4);
|
assert_eq!(entry_4.pos, IndexPosition(4));
|
||||||
assert_eq!(entry_4.generation_number(), 2);
|
assert_eq!(entry_4.generation_number(), 2);
|
||||||
assert_eq!(entry_4.num_parents(), 1);
|
assert_eq!(entry_4.num_parents(), 1);
|
||||||
assert_eq!(entry_4.parent_positions(), vec![1]);
|
assert_eq!(entry_4.parent_positions(), vec![IndexPosition(1)]);
|
||||||
assert_eq!(entry_4.num_predecessors(), 2);
|
assert_eq!(entry_4.num_predecessors(), 2);
|
||||||
assert_eq!(entry_4.predecessor_positions(), vec![2, 3]);
|
assert_eq!(
|
||||||
|
entry_4.predecessor_positions(),
|
||||||
|
vec![IndexPosition(2), IndexPosition(3)]
|
||||||
|
);
|
||||||
assert_eq!(entry_4.predecessors().len(), 2);
|
assert_eq!(entry_4.predecessors().len(), 2);
|
||||||
assert_eq!(entry_4.predecessors()[0].pos, 2);
|
assert_eq!(entry_4.predecessors()[0].pos, IndexPosition(2));
|
||||||
assert_eq!(entry_4.predecessors()[1].pos, 3);
|
assert_eq!(entry_4.predecessors()[1].pos, IndexPosition(3));
|
||||||
assert_eq!(entry_5.generation_number(), 3);
|
assert_eq!(entry_5.generation_number(), 3);
|
||||||
assert_eq!(entry_5.num_parents(), 2);
|
assert_eq!(entry_5.num_parents(), 2);
|
||||||
assert_eq!(entry_5.parent_positions(), vec![4, 2]);
|
assert_eq!(
|
||||||
|
entry_5.parent_positions(),
|
||||||
|
vec![IndexPosition(4), IndexPosition(2)]
|
||||||
|
);
|
||||||
assert_eq!(entry_5.parents().len(), 2);
|
assert_eq!(entry_5.parents().len(), 2);
|
||||||
assert_eq!(entry_5.parents()[0].pos, 4);
|
assert_eq!(entry_5.parents()[0].pos, IndexPosition(4));
|
||||||
assert_eq!(entry_5.parents()[1].pos, 2);
|
assert_eq!(entry_5.parents()[1].pos, IndexPosition(2));
|
||||||
assert_eq!(entry_5.num_predecessors(), 0);
|
assert_eq!(entry_5.num_predecessors(), 0);
|
||||||
assert_eq!(entry_5.predecessor_positions(), Vec::<u32>::new());
|
assert_eq!(entry_5.predecessor_positions(), Vec::<IndexPosition>::new());
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test_case(false; "in memory")]
|
#[test_case(false; "in memory")]
|
||||||
|
@ -1876,7 +1898,16 @@ mod tests {
|
||||||
let entry_6 = index.entry_by_id(&id_6).unwrap();
|
let entry_6 = index.entry_by_id(&id_6).unwrap();
|
||||||
assert_eq!(entry_6.commit_id(), id_6.clone());
|
assert_eq!(entry_6.commit_id(), id_6.clone());
|
||||||
assert_eq!(entry_6.num_parents(), 5);
|
assert_eq!(entry_6.num_parents(), 5);
|
||||||
assert_eq!(entry_6.parent_positions(), vec![1, 2, 3, 4, 5]);
|
assert_eq!(
|
||||||
|
entry_6.parent_positions(),
|
||||||
|
vec![
|
||||||
|
IndexPosition(1),
|
||||||
|
IndexPosition(2),
|
||||||
|
IndexPosition(3),
|
||||||
|
IndexPosition(4),
|
||||||
|
IndexPosition(5)
|
||||||
|
]
|
||||||
|
);
|
||||||
assert_eq!(entry_6.generation_number(), 2);
|
assert_eq!(entry_6.generation_number(), 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -22,7 +22,7 @@ use pest::Parser;
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
|
|
||||||
use crate::commit::Commit;
|
use crate::commit::Commit;
|
||||||
use crate::index::{HexPrefix, IndexEntry, PrefixResolution, RevWalk};
|
use crate::index::{HexPrefix, IndexEntry, IndexPosition, PrefixResolution, RevWalk};
|
||||||
use crate::repo::RepoRef;
|
use crate::repo::RepoRef;
|
||||||
use crate::store::{CommitId, StoreError};
|
use crate::store::{CommitId, StoreError};
|
||||||
|
|
||||||
|
@ -532,7 +532,7 @@ impl<'repo> Revset<'repo> for ChildrenRevset<'_, 'repo> {
|
||||||
|
|
||||||
struct ChildrenRevsetIterator<'revset, 'repo> {
|
struct ChildrenRevsetIterator<'revset, 'repo> {
|
||||||
candidate_iter: Box<dyn Iterator<Item = IndexEntry<'repo>> + 'revset>,
|
candidate_iter: Box<dyn Iterator<Item = IndexEntry<'repo>> + 'revset>,
|
||||||
roots: HashSet<u32>,
|
roots: HashSet<IndexPosition>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'repo> Iterator for ChildrenRevsetIterator<'_, 'repo> {
|
impl<'repo> Iterator for ChildrenRevsetIterator<'_, 'repo> {
|
||||||
|
|
Loading…
Reference in a new issue