diff --git a/hercules_opt/src/dce.rs b/hercules_opt/src/dce.rs
index 026672a395d783c0abd5257894c4c32335654371..6eec42e59ea102c44e4464ee981b683f988b00d8 100644
--- a/hercules_opt/src/dce.rs
+++ b/hercules_opt/src/dce.rs
@@ -8,7 +8,7 @@ use crate::*;
  */
 pub fn dce(editor: &mut FunctionEditor) {
     // Create worklist (starts as all nodes).
-    let mut worklist: Vec<NodeID> = (0..editor.func().nodes.len()).map(NodeID::new).collect();
+    let mut worklist: Vec<NodeID> = editor.node_ids().collect();
 
     while let Some(work) = worklist.pop() {
         // If a node on the worklist is a start node, it is either *the* start
diff --git a/hercules_opt/src/lib.rs b/hercules_opt/src/lib.rs
index 2c9d437238641beb9d1e609a3a288c6d02763c3e..0b10bdaef74168d99471632ffdfe08068fd9fb24 100644
--- a/hercules_opt/src/lib.rs
+++ b/hercules_opt/src/lib.rs
@@ -13,6 +13,7 @@ pub mod gcm;
 pub mod gvn;
 pub mod inline;
 pub mod interprocedural_sroa;
+pub mod lift_dc_math;
 pub mod outline;
 pub mod phi_elim;
 pub mod pred;
@@ -35,6 +36,7 @@ pub use crate::gcm::*;
 pub use crate::gvn::*;
 pub use crate::inline::*;
 pub use crate::interprocedural_sroa::*;
+pub use crate::lift_dc_math::*;
 pub use crate::outline::*;
 pub use crate::phi_elim::*;
 pub use crate::pred::*;
diff --git a/hercules_opt/src/lift_dc_math.rs b/hercules_opt/src/lift_dc_math.rs
new file mode 100644
index 0000000000000000000000000000000000000000..afdb212064d84a0191f87ce366d67b7ea6728fa8
--- /dev/null
+++ b/hercules_opt/src/lift_dc_math.rs
@@ -0,0 +1,90 @@
+use hercules_ir::ir::*;
+
+use crate::*;
+
+/*
+ * Lift math in IR nodes into dynamic constants.
+ */
+pub fn lift_dc_math(editor: &mut FunctionEditor) {
+    // Create worklist (starts as all nodes).
+    let mut worklist: Vec<NodeID> = editor.node_ids().collect();
+    while let Some(work) = worklist.pop() {
+        // Look for single nodes that can be converted to dynamic constants.
+        let users: Vec<_> = editor.get_users(work).collect();
+        let nodes = &editor.func().nodes;
+        let dc = match nodes[work.idx()] {
+            Node::Constant { id } => {
+                // Why do we need this weird crap? This is due to a limitation
+                // in Rust's lifetime rules w/ let guards.
+                let cons = if let Constant::UnsignedInteger64(cons) = *editor.get_constant(id) {
+                    cons
+                } else {
+                    continue;
+                };
+                DynamicConstant::Constant(cons as usize)
+            }
+            Node::DynamicConstant { id } => {
+                let Some(cons) = evaluate_dynamic_constant(id, &*editor.get_dynamic_constants())
+                else {
+                    continue;
+                };
+                DynamicConstant::Constant(cons)
+            }
+            Node::Binary { op, left, right } => {
+                let (left, right) = if let (
+                    Node::DynamicConstant { id: left },
+                    Node::DynamicConstant { id: right },
+                ) = (&nodes[left.idx()], &nodes[right.idx()])
+                {
+                    (*left, *right)
+                } else {
+                    continue;
+                };
+                match op {
+                    BinaryOperator::Add => DynamicConstant::Add(left, right),
+                    BinaryOperator::Sub => DynamicConstant::Sub(left, right),
+                    BinaryOperator::Mul => DynamicConstant::Mul(left, right),
+                    BinaryOperator::Div => DynamicConstant::Div(left, right),
+                    BinaryOperator::Rem => DynamicConstant::Rem(left, right),
+                    _ => {
+                        continue;
+                    }
+                }
+            }
+            Node::IntrinsicCall {
+                intrinsic,
+                ref args,
+            } => {
+                let (left, right) = if args.len() == 2
+                    && let (Node::DynamicConstant { id: left }, Node::DynamicConstant { id: right }) =
+                        (&nodes[args[0].idx()], &nodes[args[1].idx()])
+                {
+                    (*left, *right)
+                } else {
+                    continue;
+                };
+                match intrinsic {
+                    Intrinsic::Min => DynamicConstant::Min(left, right),
+                    Intrinsic::Max => DynamicConstant::Max(left, right),
+                    _ => {
+                        continue;
+                    }
+                }
+            }
+            _ => {
+                continue;
+            }
+        };
+
+        // Replace the node with the computed dynamic constant.
+        let success = editor.edit(|mut edit| {
+            let dc = edit.add_dynamic_constant(dc);
+            let node = edit.add_node(Node::DynamicConstant { id: dc });
+            edit = edit.replace_all_uses(work, node)?;
+            edit.delete_node(work)
+        });
+        if success {
+            worklist.extend(users);
+        }
+    }
+}
diff --git a/juno_scheduler/src/default.rs b/juno_scheduler/src/default.rs
index 8274b81a07e3659b130b6945ed98af9988d2575b..46b51b43b8040105f92d452df8f939522df40d2c 100644
--- a/juno_scheduler/src/default.rs
+++ b/juno_scheduler/src/default.rs
@@ -60,6 +60,10 @@ pub fn default_schedule() -> ScheduleStmt {
         DCE,
         GVN,
         DCE,
+        LiftDCMath,
+        DCE,
+        GVN,
+        DCE,
         /*Forkify,*/
         /*ForkGuardElim,*/
         DCE,
diff --git a/juno_scheduler/src/ir.rs b/juno_scheduler/src/ir.rs
index 16f2de9b1856110a8fdde8c19186a7f92f425093..381c3475e0b4f52285bf333f59cbe175d60fce60 100644
--- a/juno_scheduler/src/ir.rs
+++ b/juno_scheduler/src/ir.rs
@@ -18,6 +18,7 @@ pub enum Pass {
     InferSchedules,
     Inline,
     InterproceduralSROA,
+    LiftDCMath,
     Outline,
     PhiElim,
     Predication,
diff --git a/juno_scheduler/src/pm.rs b/juno_scheduler/src/pm.rs
index 43fba4fde2f540db625cf5012b9ec776864f1cf8..72150570d9446a33be19b8586b8092801f18dffe 100644
--- a/juno_scheduler/src/pm.rs
+++ b/juno_scheduler/src/pm.rs
@@ -6,8 +6,8 @@ use hercules_opt::FunctionEditor;
 use hercules_opt::{
     ccp, collapse_returns, crc, dce, dumb_outline, ensure_between_control_flow, float_collections,
     fork_split, gcm, gvn, infer_parallel_fork, infer_parallel_reduce, infer_tight_associative,
-    infer_vectorizable, inline, interprocedural_sroa, outline, phi_elim, predication, slf, sroa,
-    unforkify, write_predication,
+    infer_vectorizable, inline, interprocedural_sroa, lift_dc_math, outline, phi_elim, predication,
+    slf, sroa, unforkify, write_predication,
 };
 
 use tempfile::TempDir;
@@ -1339,6 +1339,18 @@ fn run_pass(
             pm.delete_gravestones();
             pm.clear_analyses();
         }
+        Pass::LiftDCMath => {
+            assert!(args.is_empty());
+            for func in build_selection(pm, selection) {
+                let Some(mut func) = func else {
+                    continue;
+                };
+                lift_dc_math(&mut func);
+                changed |= func.modified();
+            }
+            pm.delete_gravestones();
+            pm.clear_analyses();
+        }
         Pass::Outline => {
             let Some((nodes, func)) = selection_as_set(pm, selection) else {
                 return Err(SchedulerError::PassError {