Security fixes for CVE-2020-36323, CVE-2021-31162
This commit is contained in:
parent
5789d99323
commit
1cb0a34c53
15
rust.spec
15
rust.spec
@ -53,7 +53,7 @@
|
||||
|
||||
Name: rust
|
||||
Version: 1.51.0
|
||||
Release: 2%{?dist}
|
||||
Release: 3%{?dist}
|
||||
Summary: The Rust Programming Language
|
||||
License: (ASL 2.0 or MIT) and (BSD and MIT)
|
||||
# ^ written as: (rust itself) and (bundled libraries)
|
||||
@ -87,6 +87,14 @@ Patch4: rustc-1.51.0-backport-pr82292.patch
|
||||
# https://github.com/rust-lang/rust/pull/81910
|
||||
Patch5: rustc-1.51.0-backport-pr81910.patch
|
||||
|
||||
# CVE-2020-36323 rust: optimization for joining strings can cause uninitialized bytes to be exposed
|
||||
# https://github.com/rust-lang/rust/pull/81728
|
||||
Patch6: rustc-1.51.0-backport-pr81728.patch
|
||||
|
||||
# CVE-2021-31162 rust: double free in Vec::from_iter function if freeing the element panics
|
||||
# https://github.com/rust-lang/rust/pull/83629
|
||||
Patch7: rustc-1.51.0-backport-pr83629.patch
|
||||
|
||||
### RHEL-specific patches below ###
|
||||
|
||||
# Disable cargo->libgit2->libssh2 on RHEL, as it's not approved for FIPS (rhbz1732949)
|
||||
@ -425,6 +433,8 @@ test -f '%{local_rust_root}/bin/rustc'
|
||||
%patch3 -p1
|
||||
%patch4 -p1
|
||||
%patch5 -p1
|
||||
%patch6 -p1
|
||||
%patch7 -p1
|
||||
|
||||
%if %with disabled_libssh2
|
||||
%patch100 -p1
|
||||
@ -755,6 +765,9 @@ export %{rust_env}
|
||||
|
||||
|
||||
%changelog
|
||||
* Fri Apr 16 2021 Josh Stone <jistone@redhat.com> - 1.51.0-3
|
||||
- Security fixes for CVE-2020-36323, CVE-2021-31162
|
||||
|
||||
* Wed Apr 14 2021 Josh Stone <jistone@redhat.com> - 1.51.0-2
|
||||
- Security fixes for CVE-2021-28876, CVE-2021-28878, CVE-2021-28879
|
||||
- Fix bootstrap for stage0 rust 1.51
|
||||
|
181
rustc-1.51.0-backport-pr81728.patch
Normal file
181
rustc-1.51.0-backport-pr81728.patch
Normal file
@ -0,0 +1,181 @@
|
||||
From 70f17ca715d3d7e2fd79cc909b95fd3a6357c13e Mon Sep 17 00:00:00 2001
|
||||
From: Yechan Bae <yechan@gatech.edu>
|
||||
Date: Wed, 3 Feb 2021 16:36:33 -0500
|
||||
Subject: [PATCH 1/2] Fixes #80335
|
||||
|
||||
---
|
||||
library/alloc/src/str.rs | 42 ++++++++++++++++++++++----------------
|
||||
library/alloc/tests/str.rs | 30 +++++++++++++++++++++++++++
|
||||
2 files changed, 54 insertions(+), 18 deletions(-)
|
||||
|
||||
diff --git a/library/alloc/src/str.rs b/library/alloc/src/str.rs
|
||||
index 70e0c7dba5ea..a7584c6b6510 100644
|
||||
--- a/library/alloc/src/str.rs
|
||||
+++ b/library/alloc/src/str.rs
|
||||
@@ -90,8 +90,8 @@ fn join(slice: &Self, sep: &str) -> String {
|
||||
}
|
||||
}
|
||||
|
||||
-macro_rules! spezialize_for_lengths {
|
||||
- ($separator:expr, $target:expr, $iter:expr; $($num:expr),*) => {
|
||||
+macro_rules! specialize_for_lengths {
|
||||
+ ($separator:expr, $target:expr, $iter:expr; $($num:expr),*) => {{
|
||||
let mut target = $target;
|
||||
let iter = $iter;
|
||||
let sep_bytes = $separator;
|
||||
@@ -102,7 +102,8 @@ macro_rules! spezialize_for_lengths {
|
||||
$num => {
|
||||
for s in iter {
|
||||
copy_slice_and_advance!(target, sep_bytes);
|
||||
- copy_slice_and_advance!(target, s.borrow().as_ref());
|
||||
+ let content_bytes = s.borrow().as_ref();
|
||||
+ copy_slice_and_advance!(target, content_bytes);
|
||||
}
|
||||
},
|
||||
)*
|
||||
@@ -110,11 +111,13 @@ macro_rules! spezialize_for_lengths {
|
||||
// arbitrary non-zero size fallback
|
||||
for s in iter {
|
||||
copy_slice_and_advance!(target, sep_bytes);
|
||||
- copy_slice_and_advance!(target, s.borrow().as_ref());
|
||||
+ let content_bytes = s.borrow().as_ref();
|
||||
+ copy_slice_and_advance!(target, content_bytes);
|
||||
}
|
||||
}
|
||||
}
|
||||
- };
|
||||
+ target
|
||||
+ }}
|
||||
}
|
||||
|
||||
macro_rules! copy_slice_and_advance {
|
||||
@@ -153,7 +156,7 @@ fn join_generic_copy<B, T, S>(slice: &[S], sep: &[T]) -> Vec<T>
|
||||
// if the `len` calculation overflows, we'll panic
|
||||
// we would have run out of memory anyway and the rest of the function requires
|
||||
// the entire Vec pre-allocated for safety
|
||||
- let len = sep_len
|
||||
+ let reserved_len = sep_len
|
||||
.checked_mul(iter.len())
|
||||
.and_then(|n| {
|
||||
slice.iter().map(|s| s.borrow().as_ref().len()).try_fold(n, usize::checked_add)
|
||||
@@ -161,22 +164,25 @@ fn join_generic_copy<B, T, S>(slice: &[S], sep: &[T]) -> Vec<T>
|
||||
.expect("attempt to join into collection with len > usize::MAX");
|
||||
|
||||
// crucial for safety
|
||||
- let mut result = Vec::with_capacity(len);
|
||||
- assert!(result.capacity() >= len);
|
||||
+ let mut result = Vec::with_capacity(reserved_len);
|
||||
+ debug_assert!(result.capacity() >= reserved_len);
|
||||
|
||||
result.extend_from_slice(first.borrow().as_ref());
|
||||
|
||||
unsafe {
|
||||
- {
|
||||
- let pos = result.len();
|
||||
- let target = result.get_unchecked_mut(pos..len);
|
||||
-
|
||||
- // copy separator and slices over without bounds checks
|
||||
- // generate loops with hardcoded offsets for small separators
|
||||
- // massive improvements possible (~ x2)
|
||||
- spezialize_for_lengths!(sep, target, iter; 0, 1, 2, 3, 4);
|
||||
- }
|
||||
- result.set_len(len);
|
||||
+ let pos = result.len();
|
||||
+ let target = result.get_unchecked_mut(pos..reserved_len);
|
||||
+
|
||||
+ // copy separator and slices over without bounds checks
|
||||
+ // generate loops with hardcoded offsets for small separators
|
||||
+ // massive improvements possible (~ x2)
|
||||
+ let remain = specialize_for_lengths!(sep, target, iter; 0, 1, 2, 3, 4);
|
||||
+
|
||||
+ // issue #80335: A weird borrow implementation can return different
|
||||
+ // slices for the length calculation and the actual copy, so
|
||||
+ // `remain.len()` might be non-zero.
|
||||
+ let result_len = reserved_len - remain.len();
|
||||
+ result.set_len(result_len);
|
||||
}
|
||||
result
|
||||
}
|
||||
diff --git a/library/alloc/tests/str.rs b/library/alloc/tests/str.rs
|
||||
index 604835e6cc4a..6df8d8c2f354 100644
|
||||
--- a/library/alloc/tests/str.rs
|
||||
+++ b/library/alloc/tests/str.rs
|
||||
@@ -160,6 +160,36 @@ fn test_join_for_different_lengths_with_long_separator() {
|
||||
test_join!("~~~~~a~~~~~bc", ["", "a", "bc"], "~~~~~");
|
||||
}
|
||||
|
||||
+#[test]
|
||||
+fn test_join_isue_80335() {
|
||||
+ use core::{borrow::Borrow, cell::Cell};
|
||||
+
|
||||
+ struct WeirdBorrow {
|
||||
+ state: Cell<bool>,
|
||||
+ }
|
||||
+
|
||||
+ impl Default for WeirdBorrow {
|
||||
+ fn default() -> Self {
|
||||
+ WeirdBorrow { state: Cell::new(false) }
|
||||
+ }
|
||||
+ }
|
||||
+
|
||||
+ impl Borrow<str> for WeirdBorrow {
|
||||
+ fn borrow(&self) -> &str {
|
||||
+ let state = self.state.get();
|
||||
+ if state {
|
||||
+ "0"
|
||||
+ } else {
|
||||
+ self.state.set(true);
|
||||
+ "123456"
|
||||
+ }
|
||||
+ }
|
||||
+ }
|
||||
+
|
||||
+ let arr: [WeirdBorrow; 3] = Default::default();
|
||||
+ test_join!("0-0-0", arr, "-");
|
||||
+}
|
||||
+
|
||||
#[test]
|
||||
#[cfg_attr(miri, ignore)] // Miri is too slow
|
||||
fn test_unsafe_slice() {
|
||||
--
|
||||
2.31.1
|
||||
|
||||
|
||||
From 10020817d2e6756be1ff2ac3c182af97cf7fe904 Mon Sep 17 00:00:00 2001
|
||||
From: Yechan Bae <yechan@gatech.edu>
|
||||
Date: Sat, 20 Mar 2021 13:42:54 -0400
|
||||
Subject: [PATCH 2/2] Update the comment
|
||||
|
||||
---
|
||||
library/alloc/src/str.rs | 8 ++++----
|
||||
1 file changed, 4 insertions(+), 4 deletions(-)
|
||||
|
||||
diff --git a/library/alloc/src/str.rs b/library/alloc/src/str.rs
|
||||
index a7584c6b6510..4d1e876457b8 100644
|
||||
--- a/library/alloc/src/str.rs
|
||||
+++ b/library/alloc/src/str.rs
|
||||
@@ -163,7 +163,7 @@ fn join_generic_copy<B, T, S>(slice: &[S], sep: &[T]) -> Vec<T>
|
||||
})
|
||||
.expect("attempt to join into collection with len > usize::MAX");
|
||||
|
||||
- // crucial for safety
|
||||
+ // prepare an uninitialized buffer
|
||||
let mut result = Vec::with_capacity(reserved_len);
|
||||
debug_assert!(result.capacity() >= reserved_len);
|
||||
|
||||
@@ -178,9 +178,9 @@ fn join_generic_copy<B, T, S>(slice: &[S], sep: &[T]) -> Vec<T>
|
||||
// massive improvements possible (~ x2)
|
||||
let remain = specialize_for_lengths!(sep, target, iter; 0, 1, 2, 3, 4);
|
||||
|
||||
- // issue #80335: A weird borrow implementation can return different
|
||||
- // slices for the length calculation and the actual copy, so
|
||||
- // `remain.len()` might be non-zero.
|
||||
+ // A weird borrow implementation may return different
|
||||
+ // slices for the length calculation and the actual copy.
|
||||
+ // Make sure we don't expose uninitialized bytes to the caller.
|
||||
let result_len = reserved_len - remain.len();
|
||||
result.set_len(result_len);
|
||||
}
|
||||
--
|
||||
2.31.1
|
||||
|
142
rustc-1.51.0-backport-pr83629.patch
Normal file
142
rustc-1.51.0-backport-pr83629.patch
Normal file
@ -0,0 +1,142 @@
|
||||
From 3834e7b7393bf1a0d7df02ccd1d2e896c1465769 Mon Sep 17 00:00:00 2001
|
||||
From: The8472 <git@infinite-source.de>
|
||||
Date: Mon, 29 Mar 2021 04:22:34 +0200
|
||||
Subject: [PATCH 1/2] add testcase for double-drop during Vec in-place
|
||||
collection
|
||||
|
||||
---
|
||||
library/alloc/tests/vec.rs | 38 +++++++++++++++++++++++++++++++++++++-
|
||||
1 file changed, 37 insertions(+), 1 deletion(-)
|
||||
|
||||
diff --git a/library/alloc/tests/vec.rs b/library/alloc/tests/vec.rs
|
||||
index 5c7ff67bc621..4cdb7eefcdf1 100644
|
||||
--- a/library/alloc/tests/vec.rs
|
||||
+++ b/library/alloc/tests/vec.rs
|
||||
@@ -954,7 +954,7 @@ fn test_from_iter_specialization_head_tail_drop() {
|
||||
}
|
||||
|
||||
#[test]
|
||||
-fn test_from_iter_specialization_panic_drop() {
|
||||
+fn test_from_iter_specialization_panic_during_iteration_drops() {
|
||||
let drop_count: Vec<_> = (0..=2).map(|_| Rc::new(())).collect();
|
||||
let src: Vec<_> = drop_count.iter().cloned().collect();
|
||||
let iter = src.into_iter();
|
||||
@@ -977,6 +977,42 @@ fn test_from_iter_specialization_panic_drop() {
|
||||
);
|
||||
}
|
||||
|
||||
+#[test]
|
||||
+fn test_from_iter_specialization_panic_during_drop_leaks() {
|
||||
+ static mut DROP_COUNTER: usize = 0;
|
||||
+
|
||||
+ #[derive(Debug)]
|
||||
+ enum Droppable {
|
||||
+ DroppedTwice(Box<i32>),
|
||||
+ PanicOnDrop,
|
||||
+ }
|
||||
+
|
||||
+ impl Drop for Droppable {
|
||||
+ fn drop(&mut self) {
|
||||
+ match self {
|
||||
+ Droppable::DroppedTwice(_) => {
|
||||
+ unsafe {
|
||||
+ DROP_COUNTER += 1;
|
||||
+ }
|
||||
+ println!("Dropping!")
|
||||
+ }
|
||||
+ Droppable::PanicOnDrop => {
|
||||
+ if !std::thread::panicking() {
|
||||
+ panic!();
|
||||
+ }
|
||||
+ }
|
||||
+ }
|
||||
+ }
|
||||
+ }
|
||||
+
|
||||
+ let _ = std::panic::catch_unwind(AssertUnwindSafe(|| {
|
||||
+ let v = vec![Droppable::DroppedTwice(Box::new(123)), Droppable::PanicOnDrop];
|
||||
+ let _ = v.into_iter().take(0).collect::<Vec<_>>();
|
||||
+ }));
|
||||
+
|
||||
+ assert_eq!(unsafe { DROP_COUNTER }, 1);
|
||||
+}
|
||||
+
|
||||
#[test]
|
||||
fn test_cow_from() {
|
||||
let borrowed: &[_] = &["borrowed", "(slice)"];
|
||||
--
|
||||
2.31.1
|
||||
|
||||
|
||||
From 8e2706343e1ce1c5a2d3a2ceaaaa010aaeb21d93 Mon Sep 17 00:00:00 2001
|
||||
From: The8472 <git@infinite-source.de>
|
||||
Date: Mon, 29 Mar 2021 04:22:48 +0200
|
||||
Subject: [PATCH 2/2] fix double-drop in in-place collect specialization
|
||||
|
||||
---
|
||||
library/alloc/src/vec/into_iter.rs | 27 ++++++++++++++-------
|
||||
library/alloc/src/vec/source_iter_marker.rs | 4 +--
|
||||
2 files changed, 20 insertions(+), 11 deletions(-)
|
||||
|
||||
diff --git a/library/alloc/src/vec/into_iter.rs b/library/alloc/src/vec/into_iter.rs
|
||||
index f131d06bb18f..74adced53f6d 100644
|
||||
--- a/library/alloc/src/vec/into_iter.rs
|
||||
+++ b/library/alloc/src/vec/into_iter.rs
|
||||
@@ -85,20 +85,29 @@ fn as_raw_mut_slice(&mut self) -> *mut [T] {
|
||||
ptr::slice_from_raw_parts_mut(self.ptr as *mut T, self.len())
|
||||
}
|
||||
|
||||
- pub(super) fn drop_remaining(&mut self) {
|
||||
- unsafe {
|
||||
- ptr::drop_in_place(self.as_mut_slice());
|
||||
- }
|
||||
- self.ptr = self.end;
|
||||
- }
|
||||
+ /// Drops remaining elements and relinquishes the backing allocation.
|
||||
+ ///
|
||||
+ /// This is roughly equivalent to the following, but more efficient
|
||||
+ ///
|
||||
+ /// ```
|
||||
+ /// # let mut into_iter = Vec::<u8>::with_capacity(10).into_iter();
|
||||
+ /// (&mut into_iter).for_each(core::mem::drop);
|
||||
+ /// unsafe { core::ptr::write(&mut into_iter, Vec::new().into_iter()); }
|
||||
+ /// ```
|
||||
+ pub(super) fn forget_allocation_drop_remaining(&mut self) {
|
||||
+ let remaining = self.as_raw_mut_slice();
|
||||
|
||||
- /// Relinquishes the backing allocation, equivalent to
|
||||
- /// `ptr::write(&mut self, Vec::new().into_iter())`
|
||||
- pub(super) fn forget_allocation(&mut self) {
|
||||
+ // overwrite the individual fields instead of creating a new
|
||||
+ // struct and then overwriting &mut self.
|
||||
+ // this creates less assembly
|
||||
self.cap = 0;
|
||||
self.buf = unsafe { NonNull::new_unchecked(RawVec::NEW.ptr()) };
|
||||
self.ptr = self.buf.as_ptr();
|
||||
self.end = self.buf.as_ptr();
|
||||
+
|
||||
+ unsafe {
|
||||
+ ptr::drop_in_place(remaining);
|
||||
+ }
|
||||
}
|
||||
}
|
||||
|
||||
diff --git a/library/alloc/src/vec/source_iter_marker.rs b/library/alloc/src/vec/source_iter_marker.rs
|
||||
index 8c0e95559fa1..9301f7a5184e 100644
|
||||
--- a/library/alloc/src/vec/source_iter_marker.rs
|
||||
+++ b/library/alloc/src/vec/source_iter_marker.rs
|
||||
@@ -78,9 +78,9 @@ impl<T, I> SpecFromIter<T, I> for Vec<T>
|
||||
}
|
||||
|
||||
// drop any remaining values at the tail of the source
|
||||
- src.drop_remaining();
|
||||
// but prevent drop of the allocation itself once IntoIter goes out of scope
|
||||
- src.forget_allocation();
|
||||
+ // if the drop panics then we also leak any elements collected into dst_buf
|
||||
+ src.forget_allocation_drop_remaining();
|
||||
|
||||
let vec = unsafe {
|
||||
let len = dst.offset_from(dst_buf) as usize;
|
||||
--
|
||||
2.31.1
|
||||
|
Loading…
Reference in New Issue
Block a user