From a70b26555ede9ded73ec399a9cd5e612e9fc806b Mon Sep 17 00:00:00 2001
From: Connor Wood <connorwood71@gmail.com>
Date: Mon, 24 Jul 2017 09:21:53 +0100
Subject: [PATCH] Error handling in namespace path assembly

---
 src/acpi/aml/namedobj.rs          | 32 +++++++++++++++----------------
 src/acpi/aml/namespace.rs         | 32 ++++++++++++++++++++-----------
 src/acpi/aml/namespacemodifier.rs |  8 ++++----
 src/acpi/aml/termlist.rs          |  2 +-
 4 files changed, 42 insertions(+), 32 deletions(-)

diff --git a/src/acpi/aml/namedobj.rs b/src/acpi/aml/namedobj.rs
index e4833022..b836ea61 100644
--- a/src/acpi/aml/namedobj.rs
+++ b/src/acpi/aml/namedobj.rs
@@ -180,7 +180,7 @@ fn parse_def_create_bit_field(data: &[u8],
     let bit_index = parse_term_arg(&data[2 + source_buf.len..], ctx)?;
     let name = parse_name_string(&data[1 + source_buf.len + bit_index.len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     
     ctx.add_to_namespace(local_scope_string, AmlValue::BufferField {
         source_buf: Box::new(source_buf.val),
@@ -210,7 +210,7 @@ fn parse_def_create_byte_field(data: &[u8],
     let bit_index = parse_term_arg(&data[2 + source_buf.len..], ctx)?;
     let name = parse_name_string(&data[1 + source_buf.len + bit_index.len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     
     ctx.add_to_namespace(local_scope_string, AmlValue::BufferField {
         source_buf: Box::new(source_buf.val),
@@ -240,7 +240,7 @@ fn parse_def_create_word_field(data: &[u8],
     let bit_index = parse_term_arg(&data[2 + source_buf.len..], ctx)?;
     let name = parse_name_string(&data[1 + source_buf.len + bit_index.len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     
     ctx.add_to_namespace(local_scope_string, AmlValue::BufferField {
         source_buf: Box::new(source_buf.val),
@@ -270,7 +270,7 @@ fn parse_def_create_dword_field(data: &[u8],
     let bit_index = parse_term_arg(&data[2 + source_buf.len..], ctx)?;
     let name = parse_name_string(&data[1 + source_buf.len + bit_index.len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     
     ctx.add_to_namespace(local_scope_string, AmlValue::BufferField {
         source_buf: Box::new(source_buf.val),
@@ -300,7 +300,7 @@ fn parse_def_create_qword_field(data: &[u8],
     let bit_index = parse_term_arg(&data[2 + source_buf.len..], ctx)?;
     let name = parse_name_string(&data[1 + source_buf.len + bit_index.len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     
     ctx.add_to_namespace(local_scope_string, AmlValue::BufferField {
         source_buf: Box::new(source_buf.val),
@@ -331,7 +331,7 @@ fn parse_def_create_field(data: &[u8],
     let num_bits = parse_term_arg(&data[2 + source_buf.len + bit_index.len..], ctx)?;
     let name = parse_name_string(&data[2 + source_buf.len + bit_index.len + num_bits.len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     
     ctx.add_to_namespace(local_scope_string, AmlValue::BufferField {
         source_buf: Box::new(source_buf.val),
@@ -363,7 +363,7 @@ fn parse_def_data_region(data: &[u8],
     let oem_id = parse_term_arg(&data[2 + name.len + signature.len..], ctx)?;
     let oem_table_id = parse_term_arg(&data[2 + name.len + signature.len + oem_id.len..], ctx)?;
 
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
 
     ctx.add_to_namespace(local_scope_string, AmlValue::OperationRegion {
         region: RegionSpace::SystemMemory,
@@ -396,7 +396,7 @@ fn parse_def_event(data: &[u8],
 
     let name = parse_name_string(&data[2..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     ctx.add_to_namespace(local_scope_string, AmlValue::Event(0))?;
 
     Ok(AmlParseType {
@@ -421,7 +421,7 @@ fn parse_def_device(data: &[u8],
     let (pkg_length, pkg_length_len) = parse_pkg_length(&data[2..])?;
     let name = parse_name_string(&data[2 + pkg_length_len .. 2 + pkg_length], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     let mut local_ctx = AmlExecutionContext::new(local_scope_string.clone());
     
     parse_object_list(&data[2 + pkg_length_len + name.len .. 2 + pkg_length], &mut local_ctx)?;
@@ -465,7 +465,7 @@ fn parse_def_op_region(data: &[u8],
     let offset = parse_term_arg(&data[3 + name.len..], ctx)?;
     let len = parse_term_arg(&data[3 + name.len + offset.len..], ctx)?;
 
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     ctx.add_to_namespace(local_scope_string, AmlValue::OperationRegion {
         region: region,
         offset: Box::new(offset.val),
@@ -629,7 +629,7 @@ fn parse_field_element(data: &[u8],
     }
     
     let length = if let Ok(field) = parse_named_field(data, ctx) {
-        let local_scope_string = get_namespace_string(ctx.scope.clone(), AmlValue::String(field.val.name.clone()));
+        let local_scope_string = get_namespace_string(ctx.scope.clone(), AmlValue::String(field.val.name.clone()))?;
         
         ctx.add_to_namespace(local_scope_string, AmlValue::FieldUnit {
             selector: selector.clone(),
@@ -785,7 +785,7 @@ fn parse_def_method(data: &[u8],
 
     let term_list = &data[2 + pkg_len_len + name.len .. 1 + pkg_len];
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     ctx.add_to_namespace(local_scope_string, AmlValue::Method(Method {
         arg_count,
         serialized,
@@ -815,7 +815,7 @@ fn parse_def_mutex(data: &[u8],
     let flags = data[2 + name.len];
     let sync_level = flags & 0x0F;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     ctx.add_to_namespace(local_scope_string, AmlValue::Mutex((sync_level, None)))?;
 
     Ok(AmlParseType {
@@ -840,7 +840,7 @@ fn parse_def_power_res(data: &[u8],
     let (pkg_len, pkg_len_len) = parse_pkg_length(&data[2..])?;
     let name = parse_name_string(&data[2 + pkg_len_len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
 
     let system_level = data[2 + pkg_len_len + name.len];
     let resource_order: u16 = (data[3 + pkg_len_len + name.len] as u16) +
@@ -876,7 +876,7 @@ fn parse_def_processor(data: &[u8],
     let (pkg_len, pkg_len_len) = parse_pkg_length(&data[2..])?;
     let name = parse_name_string(&data[2 + pkg_len_len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
     
     let proc_id = data[2 + pkg_len_len + name.len];
     let p_blk_addr: u32 = (data[3 + pkg_len_len + name.len] as u32) +
@@ -915,7 +915,7 @@ fn parse_def_thermal_zone(data: &[u8],
     let (pkg_len, pkg_len_len) = parse_pkg_length(&data[2..])?;
     let name = parse_name_string(&data[2 + pkg_len_len .. 2 + pkg_len], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
 
     let mut local_ctx = AmlExecutionContext::new(local_scope_string.clone());    
     parse_object_list(&data[2 + pkg_len_len + name.len .. 2 + pkg_len], &mut local_ctx)?;
diff --git a/src/acpi/aml/namespace.rs b/src/acpi/aml/namespace.rs
index 8c18f1a3..4abeeda9 100644
--- a/src/acpi/aml/namespace.rs
+++ b/src/acpi/aml/namespace.rs
@@ -173,24 +173,20 @@ impl Method {
     }
 }
 
-pub fn get_namespace_string(current: String, modifier_v: AmlValue) -> String {
+pub fn get_namespace_string(current: String, modifier_v: AmlValue) -> Result<String, AmlError> {
     // TODO: Type error if modifier not string
-    let mut modifier = if let Ok(s) = modifier_v.get_as_string() {
-        s
-    } else {
-        return current;
-    };
+    let mut modifier = modifier_v.get_as_string()?;
     
     if current.len() == 0 {
-        return modifier;
+        return Ok(modifier);
     }
 
     if modifier.len() == 0 {
-        return current;
+        return Ok(current);
     }
     
     if modifier.starts_with("\\") {
-        return modifier;
+        return Ok(modifier);
     }
 
     let mut namespace = current.clone();
@@ -198,7 +194,21 @@ pub fn get_namespace_string(current: String, modifier_v: AmlValue) -> String {
     if modifier.starts_with("^") {
         while modifier.starts_with("^") {
             modifier = modifier[1..].to_string();
-            while namespace.pop() != Some('.') { }
+
+            if namespace.ends_with("\\") {
+                return Err(AmlError::AmlValueError);
+            }
+
+            loop {
+                if namespace.ends_with(".") {
+                    namespace.pop();
+                    break;
+                }
+
+                if namespace.pop() == None {
+                    return Err(AmlError::AmlValueError);
+                }
+            }
         }
     }
 
@@ -206,5 +216,5 @@ pub fn get_namespace_string(current: String, modifier_v: AmlValue) -> String {
         namespace.push('.');
     }
     
-    namespace + &modifier
+    Ok(namespace + &modifier)
 }
diff --git a/src/acpi/aml/namespacemodifier.rs b/src/acpi/aml/namespacemodifier.rs
index 059a93c7..57d1b1a7 100644
--- a/src/acpi/aml/namespacemodifier.rs
+++ b/src/acpi/aml/namespacemodifier.rs
@@ -41,8 +41,8 @@ fn parse_alias_op(data: &[u8],
     let source_name = parse_name_string(&data[1..], ctx)?;
     let alias_name = parse_name_string(&data[1 + source_name.len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), source_name.val);
-    let local_alias_string = get_namespace_string(ctx.scope.clone(), alias_name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), source_name.val)?;
+    let local_alias_string = get_namespace_string(ctx.scope.clone(), alias_name.val)?;
 
     ctx.add_to_namespace(local_scope_string, AmlValue::ObjectReference(ObjectReference::NamedObj(local_alias_string)))?;
     
@@ -67,7 +67,7 @@ fn parse_name_op(data: &[u8],
     let name = parse_name_string(&data[1..], ctx)?;
     let data_ref_obj = parse_data_ref_obj(&data[1 + name.len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val);
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val)?;
 
     ctx.add_to_namespace(local_scope_string, data_ref_obj.val)?;
     
@@ -92,7 +92,7 @@ fn parse_scope_op(data: &[u8],
     let (pkg_length, pkg_length_len) = parse_pkg_length(&data[1..])?;
     let name = parse_name_string(&data[1 + pkg_length_len..], ctx)?;
     
-    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val.clone());
+    let local_scope_string = get_namespace_string(ctx.scope.clone(), name.val.clone())?;
     let containing_scope_string = ctx.scope.clone();
     
     ctx.scope = local_scope_string;
diff --git a/src/acpi/aml/termlist.rs b/src/acpi/aml/termlist.rs
index 700fa005..8087c44b 100644
--- a/src/acpi/aml/termlist.rs
+++ b/src/acpi/aml/termlist.rs
@@ -147,7 +147,7 @@ pub fn parse_method_invocation(data: &[u8],
     }
 
     Ok(AmlParseType {
-        val: method.execute(get_namespace_string(ctx.scope.clone(), name.val), params),
+        val: method.execute(get_namespace_string(ctx.scope.clone(), name.val)?, params),
         len: current_offset
     })
 }
-- 
GitLab