diff --git a/config/vufind/Voyager.ini b/config/vufind/Voyager.ini
index 806e595f258641d623943285f525c1e66a002513..4ea47826204f5147c3b546474c2d1143d3a33912 100644
--- a/config/vufind/Voyager.ini
+++ b/config/vufind/Voyager.ini
@@ -58,3 +58,6 @@ login_field = LAST_NAME
 ;supplements = "867az"
 ; Fields to include in indexes. Default is none.
 ;indexes = "867az"
+; By default holdings are sorted by Voyager's Holdings Sort Groups. Uncomment this
+; line to disable sorting.
+;use_sort_groups = false
diff --git a/config/vufind/VoyagerRestful.ini b/config/vufind/VoyagerRestful.ini
index b4ba28facbdd44d3ddaa15a49455f67fe45796e7..8a0157980f3e4e1f190300e708f52ed8ae7b2fb2 100644
--- a/config/vufind/VoyagerRestful.ini
+++ b/config/vufind/VoyagerRestful.ini
@@ -241,3 +241,6 @@ disableAvailabilityCheckForRequestGroups = "15:19:21:32"
 ;supplements = "867az"
 ; Fields to include in indexes. Default is none.
 ;indexes = "867az"
+; By default holdings are sorted by Voyager's Holdings Sort Groups. Uncomment this
+; line to disable sorting.
+;use_sort_groups = false
diff --git a/module/VuFind/src/VuFind/ILS/Driver/Voyager.php b/module/VuFind/src/VuFind/ILS/Driver/Voyager.php
index 97649bf5d780e8786bbf4c4a11137e7f9e1d4d23..6c69c47fe0af93b10e9bb0905612f93e5b3d8d1e 100644
--- a/module/VuFind/src/VuFind/ILS/Driver/Voyager.php
+++ b/module/VuFind/src/VuFind/ILS/Driver/Voyager.php
@@ -93,6 +93,13 @@ class Voyager extends AbstractBase
      */
     protected $logger = false;
 
+    /**
+     * Whether to use holdings sort groups to sort holdings records
+     *
+     * @var bool
+     */
+    protected $useHoldingsSortGroups;
+
     /**
      * Constructor
      *
@@ -212,6 +219,10 @@ class Voyager extends AbstractBase
             );
             throw $e;
         }
+
+        $this->useHoldingsSortGroups
+            = isset($this->config['Holdings']['use_sort_groups'])
+            ? $this->config['Holdings']['use_sort_groups'] : true;
     }
 
     /**
@@ -336,6 +347,27 @@ class Voyager extends AbstractBase
         return array('available' => $available, 'otherStatuses' => $otherStatuses);
     }
 
+    /**
+     * Helper function that returns SQL for getting a sort sequence for a location
+     *
+     * @param string $locationColumn Column in the full where clause containing
+     * the column id
+     *
+     * @return string
+     */
+    protected function getItemSortSequenceSQL($locationColumn)
+    {
+        if (!$this->useHoldingsSortGroups) {
+            return '0 as SORT_SEQ';
+        }
+
+        return '(SELECT SORT_GROUP_LOCATION.SEQUENCE_NUMBER ' .
+            "FROM $this->dbName.SORT_GROUP, $this->dbName.SORT_GROUP_LOCATION " .
+            "WHERE SORT_GROUP.SORT_GROUP_DEFAULT = 'Y' " .
+            'AND SORT_GROUP_LOCATION.SORT_GROUP_ID = SORT_GROUP.SORT_GROUP_ID ' .
+            "AND SORT_GROUP_LOCATION.LOCATION_ID = $locationColumn) SORT_SEQ";
+    }
+
     /**
      * Protected support method for getStatus -- get components required for standard
      * status lookup SQL.
@@ -353,7 +385,9 @@ class Voyager extends AbstractBase
             "NVL(LOCATION.LOCATION_DISPLAY_NAME, " .
                 "LOCATION.LOCATION_NAME) as location",
             "MFHD_MASTER.DISPLAY_CALL_NO as callnumber",
-            "ITEM.TEMP_LOCATION", "ITEM.ITEM_TYPE_ID"
+            "ITEM.TEMP_LOCATION", "ITEM.ITEM_TYPE_ID",
+            "ITEM.ITEM_SEQUENCE_NUMBER",
+            $this->getItemSortSequenceSQL('ITEM.PERM_LOCATION')
         );
 
         // From
@@ -401,35 +435,43 @@ class Voyager extends AbstractBase
     protected function getStatusNoItemsSQL($id)
     {
         // Expressions
-        $sqlExpressions = array("BIB_MFHD.BIB_ID",
-                                "1 as ITEM_ID", "'N' as ON_RESERVE",
-                                "'No information available' as status",
-                                "NVL(LOCATION.LOCATION_DISPLAY_NAME, " .
-                                    "LOCATION.LOCATION_NAME) as location",
-                                "MFHD_MASTER.DISPLAY_CALL_NO as callnumber",
-                                "0 AS TEMP_LOCATION"
-                               );
+        $sqlExpressions = array(
+            "BIB_MFHD.BIB_ID",
+            "1 as ITEM_ID", "'N' as ON_RESERVE",
+            "'No information available' as status",
+            "NVL(LOCATION.LOCATION_DISPLAY_NAME, " .
+                "LOCATION.LOCATION_NAME) as location",
+            "MFHD_MASTER.DISPLAY_CALL_NO as callnumber",
+            "0 AS TEMP_LOCATION",
+            "0 as ITEM_SEQUENCE_NUMBER",
+            $this->getItemSortSequenceSQL('LOCATION.LOCATION_ID'),
+        );
 
         // From
-        $sqlFrom = array($this->dbName.".BIB_MFHD", $this->dbName.".LOCATION",
-                         $this->dbName.".MFHD_MASTER"
-                        );
+        $sqlFrom = array(
+            $this->dbName.".BIB_MFHD", $this->dbName.".LOCATION",
+            $this->dbName.".MFHD_MASTER"
+        );
 
         // Where
-        $sqlWhere = array("BIB_MFHD.BIB_ID = :id",
-                          "LOCATION.LOCATION_ID = MFHD_MASTER.LOCATION_ID",
-                          "MFHD_MASTER.MFHD_ID = BIB_MFHD.MFHD_ID",
-                          "MFHD_MASTER.SUPPRESS_IN_OPAC='N'"
-                         );
+        $sqlWhere = array(
+            "BIB_MFHD.BIB_ID = :id",
+            "LOCATION.LOCATION_ID = MFHD_MASTER.LOCATION_ID",
+            "MFHD_MASTER.MFHD_ID = BIB_MFHD.MFHD_ID",
+            "MFHD_MASTER.SUPPRESS_IN_OPAC='N'",
+            "NOT EXISTS (SELECT MFHD_ID FROM {$this->dbName}.MFHD_ITEM " .
+            "WHERE MFHD_ITEM.MFHD_ID=MFHD_MASTER.MFHD_ID)"
+        );
 
         // Bind
         $sqlBind = array(':id' => $id);
 
-        $sqlArray = array('expressions' => $sqlExpressions,
-                          'from' => $sqlFrom,
-                          'where' => $sqlWhere,
-                          'bind' => $sqlBind,
-                          );
+        $sqlArray = array(
+            'expressions' => $sqlExpressions,
+            'from' => $sqlFrom,
+            'where' => $sqlWhere,
+            'bind' => $sqlBind,
+        );
 
         return $sqlArray;
     }
@@ -456,7 +498,11 @@ class Voyager extends AbstractBase
                         ? $this->getLocationName($row['TEMP_LOCATION'])
                         : utf8_encode($row['LOCATION']),
                     'reserve' => $row['ON_RESERVE'],
-                    'callnumber' => $row['CALLNUMBER']
+                    'callnumber' => $row['CALLNUMBER'],
+                    'item_sort_seq' => $row['ITEM_SEQUENCE_NUMBER'],
+                    'sort_seq' => isset($row['SORT_SEQ'])
+                        ? $row['SORT_SEQ']
+                        : PHP_INT_MAX
                 );
             } else {
                 if (!in_array(
@@ -496,6 +542,17 @@ class Voyager extends AbstractBase
             $status[] = $current;
         }
 
+        if ($this->useHoldingsSortGroups) {
+            uksort(
+                $status,
+                function($a, $b) {
+                    return $status[$a]['sort_seq'] == $status[$b]['sort_seq']
+                        ? $status[$a]['item_sort_seq'] - $status[$b]['item_sort_seq']
+                        : $status[$a]['sort_seq'] - $status[$b]['sort_seq'];
+                }
+            );
+        }
+
         return $status;
     }
 
@@ -524,8 +581,8 @@ class Voyager extends AbstractBase
             $this->buildSqlFromArray($sqlArrayNoItems)
         );
 
-        // Loop through the possible queries and try each in turn -- the first one
-        // that yields results will cause us to break out of the loop.
+        // Loop through the possible queries and merge results.
+        $data = array();
         foreach ($possibleQueries as $sql) {
             // Execute SQL
             try {
@@ -539,13 +596,7 @@ class Voyager extends AbstractBase
                 $sqlRows[] = $row;
             }
 
-            $data = $this->getStatusData($sqlRows);
-
-            // If we found data, we can leave the foreach loop -- we don't need to
-            // try any more queries.
-            if (count($data) > 0) {
-                break;
-            }
+            $data = array_merge($data, $this->getStatusData($sqlRows));
         }
         return $this->processStatusData($data);
     }
@@ -595,7 +646,9 @@ class Voyager extends AbstractBase
             "to_char(CIRC_TRANSACTIONS.CURRENT_DUE_DATE, 'MM-DD-YY') as duedate",
             "(SELECT TO_CHAR(MAX(CIRC_TRANS_ARCHIVE.DISCHARGE_DATE), " .
             "'MM-DD-YY HH24:MI') FROM $this->dbName.CIRC_TRANS_ARCHIVE " .
-            "WHERE CIRC_TRANS_ARCHIVE.ITEM_ID = ITEM.ITEM_ID) RETURNDATE"
+            "WHERE CIRC_TRANS_ARCHIVE.ITEM_ID = ITEM.ITEM_ID) RETURNDATE",
+            "ITEM.ITEM_SEQUENCE_NUMBER",
+            $this->getItemSortSequenceSQL('ITEM.PERM_LOCATION')
         );
 
         // From
@@ -653,32 +706,37 @@ class Voyager extends AbstractBase
     protected function getHoldingNoItemsSQL($id)
     {
         // Expressions
-        $sqlExpressions = array("null as ITEM_BARCODE", "null as ITEM_ID",
-                                "MFHD_DATA.RECORD_SEGMENT", "null as ITEM_ENUM",
-                                "'N' as ON_RESERVE", "1 as ITEM_SEQUENCE_NUMBER",
-                                "'No information available' as status",
-                                "NVL(LOCATION.LOCATION_DISPLAY_NAME, " .
-                                    "LOCATION.LOCATION_NAME) as location",
-                                "MFHD_MASTER.DISPLAY_CALL_NO as callnumber",
-                                "BIB_MFHD.BIB_ID", "MFHD_MASTER.MFHD_ID",
-                                "null as duedate", "0 AS TEMP_LOCATION",
-                                "0 as PERM_LOCATION"
-                               );
+        $sqlExpressions = array(
+            "null as ITEM_BARCODE", "null as ITEM_ID",
+            "MFHD_DATA.RECORD_SEGMENT", "null as ITEM_ENUM",
+            "'N' as ON_RESERVE", "1 as ITEM_SEQUENCE_NUMBER",
+            "'No information available' as status",
+            "NVL(LOCATION.LOCATION_DISPLAY_NAME, " .
+                "LOCATION.LOCATION_NAME) as location",
+            "MFHD_MASTER.DISPLAY_CALL_NO as callnumber",
+            "BIB_MFHD.BIB_ID", "MFHD_MASTER.MFHD_ID",
+            "null as duedate", "0 AS TEMP_LOCATION",
+            "0 as PERM_LOCATION",
+            "0 as ITEM_SEQUENCE_NUMBER",
+            $this->getItemSortSequenceSQL('LOCATION.LOCATION_ID')
+        );
 
         // From
-        $sqlFrom = array($this->dbName.".BIB_MFHD", $this->dbName.".LOCATION",
-                         $this->dbName.".MFHD_MASTER", $this->dbName.".MFHD_DATA"
-                        );
+        $sqlFrom = array(
+            $this->dbName.".BIB_MFHD", $this->dbName.".LOCATION",
+            $this->dbName.".MFHD_MASTER", $this->dbName.".MFHD_DATA"
+        );
 
         // Where
-        $sqlWhere = array("BIB_MFHD.BIB_ID = :id",
-                          "LOCATION.LOCATION_ID = MFHD_MASTER.LOCATION_ID",
-                          "MFHD_MASTER.MFHD_ID = BIB_MFHD.MFHD_ID",
-                          "MFHD_DATA.MFHD_ID = BIB_MFHD.MFHD_ID",
-                          "MFHD_MASTER.SUPPRESS_IN_OPAC='N'",
-                          "NOT EXISTS (SELECT MFHD_ID FROM {$this->dbName}.MFHD_ITEM"
-                          . " WHERE MFHD_ITEM.MFHD_ID=MFHD_MASTER.MFHD_ID)"
-                         );
+        $sqlWhere = array(
+            "BIB_MFHD.BIB_ID = :id",
+            "LOCATION.LOCATION_ID = MFHD_MASTER.LOCATION_ID",
+            "MFHD_MASTER.MFHD_ID = BIB_MFHD.MFHD_ID",
+            "MFHD_DATA.MFHD_ID = BIB_MFHD.MFHD_ID",
+            "MFHD_MASTER.SUPPRESS_IN_OPAC='N'",
+            "NOT EXISTS (SELECT MFHD_ID FROM {$this->dbName}.MFHD_ITEM"
+            . " WHERE MFHD_ITEM.MFHD_ID=MFHD_MASTER.MFHD_ID)"
+        );
 
         // Order
         $sqlOrder = array("MFHD_DATA.MFHD_ID", "MFHD_DATA.SEQNUM");
@@ -686,12 +744,13 @@ class Voyager extends AbstractBase
         // Bind
         $sqlBind = array(':id' => $id);
 
-        $sqlArray = array('expressions' => $sqlExpressions,
-                          'from' => $sqlFrom,
-                          'where' => $sqlWhere,
-                          'order' => $sqlOrder,
-                          'bind' => $sqlBind,
-                          );
+        $sqlArray = array(
+            'expressions' => $sqlExpressions,
+            'from' => $sqlFrom,
+            'where' => $sqlWhere,
+            'order' => $sqlOrder,
+            'bind' => $sqlBind,
+        );
 
         return $sqlArray;
     }
@@ -952,7 +1011,11 @@ class Voyager extends AbstractBase
             'callnumber' => $sqlRow['CALLNUMBER'],
             'barcode' => $sqlRow['ITEM_BARCODE'],
             'use_unknown_message' =>
-                in_array('No information available', $sqlRow['STATUS_ARRAY'])
+                in_array('No information available', $sqlRow['STATUS_ARRAY']),
+            'item_sort_seq' => $sqlRow['ITEM_SEQUENCE_NUMBER'],
+            'sort_seq' => isset($sqlRow['SORT_SEQ'])
+                ? $sqlRow['SORT_SEQ']
+                : PHP_INT_MAX
         );
     }
 
@@ -1047,6 +1110,18 @@ class Voyager extends AbstractBase
                 $i++;
             }
         }
+
+        if ($this->useHoldingsSortGroups) {
+            usort(
+                $holding,
+                function($a, $b) {
+                    return $a['sort_seq'] == $b['sort_seq']
+                        ? $a['item_sort_seq'] - $b['item_sort_seq']
+                        : $a['sort_seq'] - $b['sort_seq'];
+                }
+            );
+        }
+
         return $holding;
     }