diff --git a/src/lib/builtins/job_control.rs b/src/lib/builtins/job_control.rs
index ed93cff358a9618b3f8cb94d85e72d71a7618c48..7ea8a41a1e774ad3cb62a1fc3cd9727081fb6bb3 100644
--- a/src/lib/builtins/job_control.rs
+++ b/src/lib/builtins/job_control.rs
@@ -74,9 +74,8 @@ pub fn jobs(shell: &mut Shell) {
 /// If the job is stopped, the job will be resumed.
 /// If multiple jobs are given, then only the last job's exit status will be returned.
 pub fn fg(shell: &mut Shell, args: &[small::String]) -> i32 {
-    fn fg_job(shell: &mut Shell, njob: u32) -> i32 {
-        if let Some(job) = shell.background_jobs().iter().nth(njob as usize).filter(|p| p.exists())
-        {
+    fn fg_job(shell: &mut Shell, njob: usize) -> i32 {
+        if let Some(job) = shell.background_jobs().iter().nth(njob).filter(|p| p.exists()) {
             // Give the bg task the foreground, and wait for it to finish. Also resume it if it
             // isn't running
             shell.set_bg_task_in_foreground(job.pid(), !job.is_running())
@@ -97,7 +96,7 @@ pub fn fg(shell: &mut Shell, args: &[small::String]) -> i32 {
         };
     } else {
         for arg in args {
-            match arg.parse::<u32>() {
+            match arg.parse::<usize>() {
                 Ok(njob) => status = fg_job(shell, njob),
                 Err(_) => {
                     eprintln!("ion: fg: {} is not a valid job number", arg);
@@ -111,9 +110,8 @@ pub fn fg(shell: &mut Shell, args: &[small::String]) -> i32 {
 
 /// Resumes a stopped background process, if it was stopped.
 pub fn bg(shell: &mut Shell, args: &[small::String]) -> i32 {
-    fn bg_job(shell: &mut Shell, njob: u32) -> bool {
-        if let Some(job) = shell.background_jobs().iter().nth(njob as usize).filter(|p| p.exists())
-        {
+    fn bg_job(shell: &mut Shell, njob: usize) -> bool {
+        if let Some(job) = shell.background_jobs().iter().nth(njob).filter(|p| p.exists()) {
             if job.is_running() {
                 eprintln!("ion: bg: job {} is already running", njob);
                 false
@@ -140,7 +138,7 @@ pub fn bg(shell: &mut Shell, args: &[small::String]) -> i32 {
         }
     } else {
         for arg in args {
-            if let Ok(njob) = arg.parse::<u32>() {
+            if let Ok(njob) = arg.parse::<usize>() {
                 if !bg_job(shell, njob) {
                     return FAILURE;
                 }
diff --git a/src/lib/shell/mod.rs b/src/lib/shell/mod.rs
index 83de75a9791f274569e2cfb1cc81b856285d830f..1856e7bae342f63328692f3ceef9f8fb3a4cf2ca 100644
--- a/src/lib/shell/mod.rs
+++ b/src/lib/shell/mod.rs
@@ -89,7 +89,7 @@ pub struct Shell<'a> {
     /// here.
     previous_status: i32,
     /// The job ID of the previous command sent to the background.
-    previous_job: u32,
+    previous_job: usize,
     /// Contains all the options relative to the shell
     opts: ShellOptions,
     /// Contains information on all of the active background processes that are being managed
@@ -381,7 +381,7 @@ impl<'a> Shell<'a> {
         exit_status
     }
 
-    pub fn previous_job(&self) -> Option<u32> {
+    pub fn previous_job(&self) -> Option<usize> {
         if self.previous_job == !0 {
             None
         } else {
diff --git a/src/lib/shell/pipe_exec/fork.rs b/src/lib/shell/pipe_exec/fork.rs
index e30b363de6f318aadbe48869f7a8769fd686383e..39929969d1323432a9258832b932b6b82b73f81a 100644
--- a/src/lib/shell/pipe_exec/fork.rs
+++ b/src/lib/shell/pipe_exec/fork.rs
@@ -2,7 +2,7 @@ use crate::sys;
 
 use super::{
     super::{status::*, Shell},
-    job_control::ProcessState,
+    job_control::{BackgroundProcess, ProcessState},
 };
 use crate::parser::pipelines::Pipeline;
 use std::process::exit;
@@ -36,7 +36,7 @@ impl<'a> Shell<'a> {
             Ok(pid) => {
                 if state != ProcessState::Empty {
                     // The parent process should add the child fork's PID to the background.
-                    self.send_to_background(pid, state, command_name);
+                    self.send_to_background(BackgroundProcess::new(pid, state, command_name));
                 }
                 SUCCESS
             }
diff --git a/src/lib/shell/pipe_exec/job_control.rs b/src/lib/shell/pipe_exec/job_control.rs
index e1df8b944143d0f2a68ca4c0b1aeb4c3d8e2117a..f1f873d6105d78cfd21af0231414f8fb974d4105 100644
--- a/src/lib/shell/pipe_exec/job_control.rs
+++ b/src/lib/shell/pipe_exec/job_control.rs
@@ -8,7 +8,7 @@ use crate::sys::{
 };
 use std::{
     fmt, process,
-    sync::{Arc, Mutex},
+    sync::Mutex,
     thread::{sleep, spawn},
     time::Duration,
 };
@@ -46,6 +46,10 @@ pub struct BackgroundProcess {
 }
 
 impl BackgroundProcess {
+    pub(super) fn new(pid: u32, state: ProcessState, name: String) -> Self {
+        BackgroundProcess { pid, ignore_sighup: false, state, name }
+    }
+
     pub fn pid(&self) -> u32 { self.pid }
 
     pub fn is_running(&self) -> bool { self.state == ProcessState::Running }
@@ -77,52 +81,40 @@ impl<'a> Shell<'a> {
         }
     }
 
-    fn add_to_background(
-        processes: &Arc<Mutex<Vec<BackgroundProcess>>>,
-        pid: u32,
-        state: ProcessState,
-        command: String,
-    ) -> u32 {
-        let mut processes = processes.lock().unwrap();
-        match (*processes).iter().position(|x| x.state == ProcessState::Empty) {
+    fn add_to_background(&mut self, job: BackgroundProcess) -> usize {
+        let mut processes = self.background_jobs_mut();
+        match processes.iter().position(|x| !x.exists()) {
             Some(id) => {
-                (*processes)[id] =
-                    BackgroundProcess { pid, ignore_sighup: false, state, name: command };
-                id as u32
+                processes[id] = job;
+                id
             }
             None => {
-                let njobs = (*processes).len();
-                (*processes).push(BackgroundProcess {
-                    pid,
-                    ignore_sighup: false,
-                    state,
-                    name: command,
-                });
-                njobs as u32
+                let njobs = processes.len();
+                processes.push(job);
+                njobs
             }
         }
     }
 
     fn watch_background(
-        fg: &Arc<ForegroundSignals>,
-        processes: &Arc<Mutex<Vec<BackgroundProcess>>>,
+        fg: &ForegroundSignals,
+        processes: &Mutex<Vec<BackgroundProcess>>,
         pgid: u32,
         njob: usize,
     ) {
-        let (mut fg_was_grabbed, mut status);
         let mut exit_status = 0;
 
         macro_rules! get_process {
             (| $ident:ident | $func:expr) => {
                 let mut processes = processes.lock().unwrap();
-                let $ident = &mut processes.iter_mut().nth(njob).unwrap();
+                let $ident = &mut processes.get_mut(njob).unwrap();
                 $func
             };
         }
 
         loop {
-            fg_was_grabbed = fg.was_grabbed(pgid);
-            status = 0;
+            let fg_was_grabbed = fg.was_grabbed(pgid);
+            let mut status = 0;
             match waitpid(-(pgid as i32), &mut status, OPTS) {
                 Err(errno) if errno == ECHILD => {
                     if !fg_was_grabbed {
@@ -130,7 +122,7 @@ impl<'a> Shell<'a> {
                     }
 
                     get_process!(|process| {
-                        process.state = ProcessState::Empty;
+                        process.forget();
                         if fg_was_grabbed {
                             fg.reply_with(exit_status as i8);
                         }
@@ -142,7 +134,7 @@ impl<'a> Shell<'a> {
                     eprintln!("ion: ([{}] {}) errored: {}", njob, pgid, errno);
 
                     get_process!(|process| {
-                        process.state = ProcessState::Empty;
+                        process.forget();
                         if fg_was_grabbed {
                             fg.errored();
                         }
@@ -177,18 +169,18 @@ impl<'a> Shell<'a> {
         }
     }
 
-    pub fn send_to_background(&mut self, pid: u32, state: ProcessState, command: String) {
-        // Increment the `Arc` counters so that these fields can be moved into
-        // the upcoming background thread.
-        let processes = self.background.clone();
-        let fg_signals = self.foreground_signals.clone();
-
+    pub fn send_to_background(&mut self, process: BackgroundProcess) {
         // Add the process to the background list, and mark the job's ID as
         // the previous job in the shell (in case fg/bg is executed w/ no args).
-        let njob = Self::add_to_background(&processes, pid, state, command);
+        let pid = process.pid();
+        let njob = self.add_to_background(process);
         self.previous_job = njob;
         eprintln!("ion: bg [{}] {}", njob, pid);
 
+        // Increment the `Arc` counters so that these fields can be moved into
+        // the upcoming background thread.
+        let processes = self.background.clone();
+        let fg_signals = self.foreground_signals.clone();
         // Spawn a background thread that will monitor the progress of the
         // background process, updating it's state changes until it finally
         // exits.
@@ -197,37 +189,27 @@ impl<'a> Shell<'a> {
 
     /// Send a kill signal to all running background tasks.
     pub fn background_send(&self, signal: i32) {
-        if signal == sys::SIGHUP {
-            for process in self.background.lock().unwrap().iter() {
-                if !process.ignore_sighup {
-                    let _ = sys::killpg(process.pid, signal);
-                }
-            }
-        } else {
-            for process in self.background.lock().unwrap().iter() {
-                if let ProcessState::Running = process.state {
-                    let _ = sys::killpg(process.pid, signal);
-                }
-            }
-        }
+        let filter: fn(&&BackgroundProcess) -> bool =
+            if signal == sys::SIGHUP { |p| !p.ignore_sighup } else { |p| p.is_running() };
+        self.background.lock().unwrap().iter().filter(filter).for_each(|p| {
+            let _ = sys::killpg(p.pid(), signal);
+        })
     }
 
     /// Resumes all stopped background jobs
     pub fn resume_stopped(&mut self) {
-        for process in self.background.lock().unwrap().iter() {
-            if process.state == ProcessState::Stopped {
-                signals::resume(process.pid);
-            }
+        for process in self.background_jobs().iter().filter(|p| p.state == ProcessState::Stopped) {
+            signals::resume(process.pid());
         }
     }
 
-    pub fn watch_foreground<T: Into<String>>(&mut self, pid: i32, command: T) -> i32 {
+    pub fn watch_foreground(&mut self, pgid: u32) -> i32 {
         let mut signaled = 0;
         let mut exit_status = 0;
 
         loop {
             let mut status = 0;
-            match waitpid(pid, &mut status, WUNTRACED) {
+            match waitpid(-(pgid as i32), &mut status, WUNTRACED) {
                 Err(errno) => match errno {
                     ECHILD if signaled == 0 => break exit_status,
                     ECHILD => break signaled,
@@ -258,11 +240,11 @@ impl<'a> Shell<'a> {
                     }
                 }
                 Ok(pid) if wifstopped(status) => {
-                    self.send_to_background(
+                    self.send_to_background(BackgroundProcess::new(
                         pid.abs() as u32,
                         ProcessState::Stopped,
-                        command.into(),
-                    );
+                        "".to_string(),
+                    ));
                     self.break_flow = true;
                     break 128 + wstopsig(status);
                 }
diff --git a/src/lib/shell/pipe_exec/mod.rs b/src/lib/shell/pipe_exec/mod.rs
index 6b48525c22ecec7cd7dda0208213ec1878db11ea..358c07ef2d7d7363449c678a58db059980c92d2c 100644
--- a/src/lib/shell/pipe_exec/mod.rs
+++ b/src/lib/shell/pipe_exec/mod.rs
@@ -204,7 +204,7 @@ impl<'b> Shell<'b> {
                 let _ = sys::tcsetpgrp(0, pid);
                 let _ = wait_for_interrupt(pid);
                 let _ = sys::kill(pid, sys::SIGCONT);
-                self.watch_foreground(-(pid as i32), "")
+                self.watch_foreground(pid)
             }
             Err(ref err) if err.kind() == io::ErrorKind::NotFound => {
                 self.command_not_found(name);
@@ -467,7 +467,7 @@ impl<'b> Shell<'b> {
                 // Waits for all of the children of the assigned pgid to finish executing,
                 // returning the exit status of the last process in the queue.
                 // Watch the foreground group, dropping all commands that exit as they exit.
-                let status = self.watch_foreground(-(pgid as i32), "");
+                let status = self.watch_foreground(pgid);
                 if status == TERMINATED {
                     if let Err(why) = sys::killpg(pgid, sys::SIGTERM) {
                         eprintln!("ion: failed to terminate foreground jobs: {}", why);