-
Notifications
You must be signed in to change notification settings - Fork 354
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Auto merge of #3101 - eduardosm:x86-aes-intrinsics, r=RalfJung
Implement `llvm.x86.aesni.*` intrinsics
- Loading branch information
Showing
5 changed files
with
539 additions
and
0 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,168 @@ | ||
use rustc_middle::ty::layout::LayoutOf as _; | ||
use rustc_middle::ty::Ty; | ||
use rustc_span::Symbol; | ||
use rustc_target::spec::abi::Abi; | ||
|
||
use crate::*; | ||
use shims::foreign_items::EmulateForeignItemResult; | ||
|
||
impl<'mir, 'tcx: 'mir> EvalContextExt<'mir, 'tcx> for crate::MiriInterpCx<'mir, 'tcx> {} | ||
pub(super) trait EvalContextExt<'mir, 'tcx: 'mir>: | ||
crate::MiriInterpCxExt<'mir, 'tcx> | ||
{ | ||
fn emulate_x86_aesni_intrinsic( | ||
&mut self, | ||
link_name: Symbol, | ||
abi: Abi, | ||
args: &[OpTy<'tcx, Provenance>], | ||
dest: &PlaceTy<'tcx, Provenance>, | ||
) -> InterpResult<'tcx, EmulateForeignItemResult> { | ||
let this = self.eval_context_mut(); | ||
// Prefix should have already been checked. | ||
let unprefixed_name = link_name.as_str().strip_prefix("llvm.x86.aesni.").unwrap(); | ||
|
||
match unprefixed_name { | ||
// Used to implement the _mm_aesdec_si128, _mm256_aesdec_epi128 | ||
// and _mm512_aesdec_epi128 functions. | ||
// Performs one round of an AES decryption on each 128-bit word of | ||
// `state` with the corresponding 128-bit key of `key`. | ||
// https://www.intel.com/content/www/us/en/docs/intrinsics-guide/index.html#text=_mm_aesdec_si128 | ||
"aesdec" | "aesdec.256" | "aesdec.512" => { | ||
let [state, key] = | ||
this.check_shim(abi, Abi::C { unwind: false }, link_name, args)?; | ||
|
||
aes_round(this, state, key, dest, |state, key| { | ||
let key = aes::Block::from(key.to_le_bytes()); | ||
let mut state = aes::Block::from(state.to_le_bytes()); | ||
// `aes::hazmat::equiv_inv_cipher_round` documentation states that | ||
// it performs the same operation as the x86 aesdec instruction. | ||
aes::hazmat::equiv_inv_cipher_round(&mut state, &key); | ||
u128::from_le_bytes(state.into()) | ||
})?; | ||
} | ||
// Used to implement the _mm_aesdeclast_si128, _mm256_aesdeclast_epi128 | ||
// and _mm512_aesdeclast_epi128 functions. | ||
// Performs last round of an AES decryption on each 128-bit word of | ||
// `state` with the corresponding 128-bit key of `key`. | ||
// https://www.intel.com/content/www/us/en/docs/intrinsics-guide/index.html#text=_mm_aesdeclast_si128 | ||
"aesdeclast" | "aesdeclast.256" | "aesdeclast.512" => { | ||
let [state, key] = | ||
this.check_shim(abi, Abi::C { unwind: false }, link_name, args)?; | ||
|
||
aes_round(this, state, key, dest, |state, key| { | ||
let mut state = aes::Block::from(state.to_le_bytes()); | ||
// `aes::hazmat::equiv_inv_cipher_round` does the following operations: | ||
// state = InvShiftRows(state) | ||
// state = InvSubBytes(state) | ||
// state = InvMixColumns(state) | ||
// state = state ^ key | ||
// But we need to skip the InvMixColumns. | ||
// First, use a zeroed key to skip the XOR. | ||
aes::hazmat::equiv_inv_cipher_round(&mut state, &aes::Block::from([0; 16])); | ||
// Then, undo the InvMixColumns with MixColumns. | ||
aes::hazmat::mix_columns(&mut state); | ||
// Finally, do the XOR. | ||
u128::from_le_bytes(state.into()) ^ key | ||
})?; | ||
} | ||
// Used to implement the _mm_aesenc_si128, _mm256_aesenc_epi128 | ||
// and _mm512_aesenc_epi128 functions. | ||
// Performs one round of an AES encryption on each 128-bit word of | ||
// `state` with the corresponding 128-bit key of `key`. | ||
// https://www.intel.com/content/www/us/en/docs/intrinsics-guide/index.html#text=_mm_aesenc_si128 | ||
"aesenc" | "aesenc.256" | "aesenc.512" => { | ||
let [state, key] = | ||
this.check_shim(abi, Abi::C { unwind: false }, link_name, args)?; | ||
|
||
aes_round(this, state, key, dest, |state, key| { | ||
let key = aes::Block::from(key.to_le_bytes()); | ||
let mut state = aes::Block::from(state.to_le_bytes()); | ||
// `aes::hazmat::cipher_round` documentation states that | ||
// it performs the same operation as the x86 aesenc instruction. | ||
aes::hazmat::cipher_round(&mut state, &key); | ||
u128::from_le_bytes(state.into()) | ||
})?; | ||
} | ||
// Used to implement the _mm_aesenclast_si128, _mm256_aesenclast_epi128 | ||
// and _mm512_aesenclast_epi128 functions. | ||
// Performs last round of an AES encryption on each 128-bit word of | ||
// `state` with the corresponding 128-bit key of `key`. | ||
// https://www.intel.com/content/www/us/en/docs/intrinsics-guide/index.html#text=_mm_aesenclast_si128 | ||
"aesenclast" | "aesenclast.256" | "aesenclast.512" => { | ||
let [state, key] = | ||
this.check_shim(abi, Abi::C { unwind: false }, link_name, args)?; | ||
|
||
aes_round(this, state, key, dest, |state, key| { | ||
let mut state = aes::Block::from(state.to_le_bytes()); | ||
// `aes::hazmat::cipher_round` does the following operations: | ||
// state = ShiftRows(state) | ||
// state = SubBytes(state) | ||
// state = MixColumns(state) | ||
// state = state ^ key | ||
// But we need to skip the MixColumns. | ||
// First, use a zeroed key to skip the XOR. | ||
aes::hazmat::cipher_round(&mut state, &aes::Block::from([0; 16])); | ||
// Then, undo the MixColumns with InvMixColumns. | ||
aes::hazmat::inv_mix_columns(&mut state); | ||
// Finally, do the XOR. | ||
u128::from_le_bytes(state.into()) ^ key | ||
})?; | ||
} | ||
// Used to implement the _mm_aesimc_si128 function. | ||
// Performs the AES InvMixColumns operation on `op` | ||
"aesimc" => { | ||
let [op] = this.check_shim(abi, Abi::C { unwind: false }, link_name, args)?; | ||
|
||
// Transmute to `u128` | ||
let op = op.transmute(this.machine.layouts.u128, this)?; | ||
let dest = dest.transmute(this.machine.layouts.u128, this)?; | ||
|
||
let state = this.read_scalar(&op)?.to_u128()?; | ||
let mut state = aes::Block::from(state.to_le_bytes()); | ||
aes::hazmat::inv_mix_columns(&mut state); | ||
|
||
this.write_scalar(Scalar::from_u128(u128::from_le_bytes(state.into())), &dest)?; | ||
} | ||
// TODO: Implement the `llvm.x86.aesni.aeskeygenassist` when possible | ||
// with an external crate. | ||
_ => return Ok(EmulateForeignItemResult::NotSupported), | ||
} | ||
Ok(EmulateForeignItemResult::NeedsJumping) | ||
} | ||
} | ||
|
||
// Performs an AES round (given by `f`) on each 128-bit word of | ||
// `state` with the corresponding 128-bit key of `key`. | ||
fn aes_round<'tcx>( | ||
this: &mut crate::MiriInterpCx<'_, 'tcx>, | ||
state: &OpTy<'tcx, Provenance>, | ||
key: &OpTy<'tcx, Provenance>, | ||
dest: &PlaceTy<'tcx, Provenance>, | ||
f: impl Fn(u128, u128) -> u128, | ||
) -> InterpResult<'tcx, ()> { | ||
assert_eq!(dest.layout.size, state.layout.size); | ||
assert_eq!(dest.layout.size, key.layout.size); | ||
|
||
// Transmute arguments to arrays of `u128`. | ||
assert_eq!(dest.layout.size.bytes() % 16, 0); | ||
let len = dest.layout.size.bytes() / 16; | ||
|
||
let u128_array_layout = | ||
this.layout_of(Ty::new_array(this.tcx.tcx, this.tcx.types.u128, len))?; | ||
|
||
let state = state.transmute(u128_array_layout, this)?; | ||
let key = key.transmute(u128_array_layout, this)?; | ||
let dest = dest.transmute(u128_array_layout, this)?; | ||
|
||
for i in 0..len { | ||
let state = this.read_scalar(&this.project_index(&state, i)?)?.to_u128()?; | ||
let key = this.read_scalar(&this.project_index(&key, i)?)?.to_u128()?; | ||
let dest = this.project_index(&dest, i)?; | ||
|
||
let res = f(state, key); | ||
|
||
this.write_scalar(Scalar::from_u128(res), &dest)?; | ||
} | ||
|
||
Ok(()) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.