Make nested arrays work
This commit is contained in:
parent
5d06ecb874
commit
86eab29173
@ -349,11 +349,7 @@ impl InstructionValue {
|
|||||||
Load(_, ty) => Ok(ty.clone()),
|
Load(_, ty) => Ok(ty.clone()),
|
||||||
Store(_, value) => value.get_type(builder),
|
Store(_, value) => value.get_type(builder),
|
||||||
ArrayAlloca(ty, _) => Ok(Type::Ptr(Box::new(ty.clone()))),
|
ArrayAlloca(ty, _) => Ok(Type::Ptr(Box::new(ty.clone()))),
|
||||||
GetElemPtr(arr, _) => match arr.get_type(builder) {
|
GetElemPtr(ptr, _) => ptr.get_type(builder),
|
||||||
Ok(Type::Ptr(elem_t)) => Ok(Type::Ptr(Box::new(*elem_t))),
|
|
||||||
Ok(_) => Err(()),
|
|
||||||
Err(_) => Err(()),
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -352,10 +352,6 @@ impl InstructionHolder {
|
|||||||
ArrayAlloca(ty, len) => {
|
ArrayAlloca(ty, len) => {
|
||||||
let array_len = ConstValue::U16(*len as u16).as_llvm(module.context_ref);
|
let array_len = ConstValue::U16(*len as u16).as_llvm(module.context_ref);
|
||||||
let array_ty = Type::Ptr(Box::new(ty.clone()));
|
let array_ty = Type::Ptr(Box::new(ty.clone()));
|
||||||
dbg!(
|
|
||||||
&ty.as_llvm(module.context_ref),
|
|
||||||
&array_ty.as_llvm(module.context_ref)
|
|
||||||
);
|
|
||||||
LLVMBuildArrayAlloca(
|
LLVMBuildArrayAlloca(
|
||||||
module.builder_ref,
|
module.builder_ref,
|
||||||
ty.as_llvm(module.context_ref),
|
ty.as_llvm(module.context_ref),
|
||||||
@ -371,12 +367,13 @@ impl InstructionHolder {
|
|||||||
.iter()
|
.iter()
|
||||||
.map(|idx| ConstValue::U32(*idx).as_llvm(module.context_ref))
|
.map(|idx| ConstValue::U32(*idx).as_llvm(module.context_ref))
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
LLVMBuildGEP2(
|
LLVMBuildGEP2(
|
||||||
module.builder_ref,
|
module.builder_ref,
|
||||||
elem_t.as_llvm(module.context_ref),
|
elem_t.as_llvm(module.context_ref),
|
||||||
module.values.get(arr).unwrap().value_ref,
|
module.values.get(arr).unwrap().value_ref,
|
||||||
indices.as_mut_ptr(),
|
indices.as_mut_ptr(),
|
||||||
1,
|
indices.len() as u32,
|
||||||
c"array_gep".as_ptr(),
|
c"array_gep".as_ptr(),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -154,33 +154,22 @@ impl<'ctx, 'a> Scope<'ctx, 'a> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl IndexedVariableReference {
|
impl IndexedVariableReference {
|
||||||
fn get_stack_value(&self, scope: &mut Scope) -> Option<StackValue> {
|
fn get_stack_value(&self, scope: &mut Scope) -> Option<(StackValue, Vec<u32>)> {
|
||||||
use StackValueKind as Kind;
|
use StackValueKind as Kind;
|
||||||
|
|
||||||
match &self.kind {
|
match &self.kind {
|
||||||
mir::IndexedVariableReferenceKind::Named(NamedVariableRef(_, name, _)) => {
|
mir::IndexedVariableReferenceKind::Named(NamedVariableRef(_, name, _)) => scope
|
||||||
scope.stack_values.get(name).cloned()
|
.stack_values
|
||||||
}
|
.get(name)
|
||||||
|
.cloned()
|
||||||
|
.map(|v| (v, Vec::new())),
|
||||||
mir::IndexedVariableReferenceKind::Index(inner, idx) => {
|
mir::IndexedVariableReferenceKind::Index(inner, idx) => {
|
||||||
let inner_val = inner.get_stack_value(scope)?;
|
let (inner_val, mut indices) = inner.get_stack_value(scope)?;
|
||||||
let inner_instr = match inner_val.0 {
|
|
||||||
Kind::Immutable(val) => val,
|
|
||||||
Kind::Mutable(val) => val,
|
|
||||||
};
|
|
||||||
|
|
||||||
match inner_val.1 {
|
match &inner_val.1 {
|
||||||
Type::Ptr(inner_ty) => {
|
Type::Ptr(_) => {
|
||||||
let gep_instr = scope
|
indices.push(*idx as u32);
|
||||||
.block
|
Some((inner_val, indices))
|
||||||
.build(Instr::GetElemPtr(inner_instr, vec![*idx as u32]))
|
|
||||||
.unwrap();
|
|
||||||
Some(StackValue(
|
|
||||||
match inner_val.0 {
|
|
||||||
Kind::Immutable(_) => Kind::Immutable(gep_instr),
|
|
||||||
Kind::Mutable(_) => Kind::Mutable(gep_instr),
|
|
||||||
},
|
|
||||||
*inner_ty,
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
_ => panic!("Tried to codegen indexing a non-indexable value!"),
|
_ => panic!("Tried to codegen indexing a non-indexable value!"),
|
||||||
}
|
}
|
||||||
@ -217,12 +206,24 @@ impl mir::Statement {
|
|||||||
None
|
None
|
||||||
}
|
}
|
||||||
mir::StmtKind::Set(var, val) => {
|
mir::StmtKind::Set(var, val) => {
|
||||||
if let Some(StackValue(kind, _)) = var.get_stack_value(scope) {
|
if let Some((StackValue(kind, mut ty), indices)) = var.get_stack_value(scope) {
|
||||||
match kind {
|
match kind {
|
||||||
StackValueKind::Immutable(_) => {
|
StackValueKind::Immutable(_) => {
|
||||||
panic!("Tried to mutate an immutable variable")
|
panic!("Tried to mutate an immutable variable")
|
||||||
}
|
}
|
||||||
StackValueKind::Mutable(ptr) => {
|
StackValueKind::Mutable(mut ptr) => {
|
||||||
|
for (i, idx) in indices.iter().enumerate() {
|
||||||
|
let Type::Ptr(inner) = ty else { panic!() };
|
||||||
|
ty = *inner;
|
||||||
|
|
||||||
|
ptr = scope
|
||||||
|
.block
|
||||||
|
.build(Instr::GetElemPtr(ptr, vec![*idx]))
|
||||||
|
.unwrap();
|
||||||
|
if i < (indices.len() - 1) {
|
||||||
|
ptr = scope.block.build(Instr::Load(ptr, ty.clone())).unwrap()
|
||||||
|
}
|
||||||
|
}
|
||||||
let expression = val.codegen(scope).unwrap();
|
let expression = val.codegen(scope).unwrap();
|
||||||
Some(scope.block.build(Instr::Store(ptr, expression)).unwrap())
|
Some(scope.block.build(Instr::Store(ptr, expression)).unwrap())
|
||||||
}
|
}
|
||||||
|
38
test.ll
Normal file
38
test.ll
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
; ModuleID = 'test'
|
||||||
|
source_filename = "test"
|
||||||
|
target datalayout = "e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:128-n8:16:32:64-S128"
|
||||||
|
target triple = "x86_64-unknown-linux-gnu"
|
||||||
|
|
||||||
|
define ptr @array() {
|
||||||
|
array:
|
||||||
|
%array_alloca = alloca i16, i16 4, align 2
|
||||||
|
%array_gep = getelementptr i16, ptr %array_alloca, i32 0
|
||||||
|
store i16 10, ptr %array_gep, align 2
|
||||||
|
%array_gep1 = getelementptr i16, ptr %array_alloca, i32 1
|
||||||
|
store i16 15, ptr %array_gep1, align 2
|
||||||
|
%array_gep2 = getelementptr i16, ptr %array_alloca, i32 2
|
||||||
|
store i16 7, ptr %array_gep2, align 2
|
||||||
|
%array_gep3 = getelementptr i16, ptr %array_alloca, i32 3
|
||||||
|
store i16 9, ptr %array_gep3, align 2
|
||||||
|
|
||||||
|
%array_alloca4 = alloca ptr, i16 1, align 8
|
||||||
|
%array_gep5 = getelementptr ptr, ptr %array_alloca4, i32 0
|
||||||
|
store ptr %array_alloca, ptr %array_gep5, align 8
|
||||||
|
ret ptr %array_alloca4
|
||||||
|
}
|
||||||
|
|
||||||
|
define i16 @main() {
|
||||||
|
main:
|
||||||
|
%call = call ptr @array()
|
||||||
|
%array_gep = getelementptr ptr, ptr %call, i32 0
|
||||||
|
%load5 = load ptr, ptr %array_gep2, align 8
|
||||||
|
%array_gep1 = getelementptr i16, ptr %load5, i32 3
|
||||||
|
store i16 5, ptr %array_gep1, align 2
|
||||||
|
|
||||||
|
%array_gep2 = getelementptr ptr, ptr %call, i32 0
|
||||||
|
%load = load ptr, ptr %array_gep2, align 8
|
||||||
|
%array_gep3 = getelementptr i16, ptr %load, i32 3
|
||||||
|
%load4 = load i16, ptr %array_gep3, align 2
|
||||||
|
|
||||||
|
ret i16 %load4
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user