+2020-12-02  Andrew Burgess  <andrew.burgess@embecosm.com>
+
+       * riscv-tdep.c (riscv_is_unknown_csr): New function,
+       implementation moved from riscv_register_reggroup_p.
+       (riscv_register_reggroup_p): Update group handling for unknown
+       CSRs.
+
 2020-12-01  Sergio Durigan Junior  <sergiodj@sergiodj.net>
 
        * dwarf2/read.c (dwz_search_other_debugdirs): New function.
 
     }
 }
 
+/* Return true if REGNUM is an unknown CSR identified in
+   riscv_tdesc_unknown_reg for GDBARCH.  */
+
+static bool
+riscv_is_unknown_csr (struct gdbarch *gdbarch, int regnum)
+{
+  struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
+  return (regnum >= tdep->unknown_csrs_first_regnum
+         && regnum < (tdep->unknown_csrs_first_regnum
+                      + tdep->unknown_csrs_count));
+}
+
 /* Implement the register_reggroup_p gdbarch method.  Is REGNUM a member
    of REGGROUP?  */
 
     {
       /* Any extra registers from the CSR tdesc_feature (identified in
         riscv_tdesc_unknown_reg) are removed from the save/restore groups
-        as some targets (QEMU) report CSRs which then can't be read.  */
-      struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
-      if ((reggroup == restore_reggroup || reggroup == save_reggroup)
-         && regnum >= tdep->unknown_csrs_first_regnum
-         && regnum < (tdep->unknown_csrs_first_regnum
-                      + tdep->unknown_csrs_count))
-       return 0;
+        as some targets (QEMU) report CSRs which then can't be read and
+        having unreadable registers in the save/restore group breaks
+        things like inferior calls.
+
+        The unknown CSRs are also removed from the general group, and
+        added into both the csr and system group.  This is inline with the
+        known CSRs (see below).  */
+      if (riscv_is_unknown_csr (gdbarch, regnum))
+       {
+         if (reggroup == restore_reggroup || reggroup == save_reggroup
+              || reggroup == general_reggroup)
+           return 0;
+         else if (reggroup == system_reggroup || reggroup == csr_reggroup)
+           return 1;
+       }
 
       /* This is some other unknown register from the target description.
         In this case we trust whatever the target description says about
 
+2020-12-02  Andrew Burgess  <andrew.burgess@embecosm.com>
+
+       * gdb.arch/riscv-tdesc-regs.exp (get_expected_result): New proc,
+       update test to use this.
+
 2020-12-01  Simon Marchi  <simon.marchi@polymtl.ca>
 
        * gdb.threads/non-ldr-exc-1.exp: Fix indentation.
 
 gdb_test "info registers \$dscratch0" "dscratch0\[ \t\]+.*"
 gdb_test "info registers \$dscratch" "dscratch\[ \t\]+.*"
 
-foreach rgroup {x_all all save restore} {
+# Return the number of times REGISTER should appear in GROUP, this
+# will either be 0 or 1.
+proc get_expected_result { register group } {
+
+    # Everything should appear once in the 'all' group.
+    if { $group == "all" || $group == "x_all" } {
+       return 1
+    }
+
+    if { $group == "save" || $group == "restore" } {
+       # Everything is in the save/restore groups except these two.
+       if { $register == "unknown_csr" || $register == "dscratch" } {
+           return 0
+       }
+       return 1
+    }
+
+    if { $group == "system" || $group == "csr" } {
+       # All the registers we check should be in these groups.
+       return 1
+    }
+
+    return 0
+}
+
+foreach rgroup {x_all all save restore general system csr} {
     # Now use 'info registers all' to see how many times the floating
     # point status registers show up in the output.
     array set reg_counts {}
        } else {
            set count 0
        }
-       if {($reg == "unknown_csr" || $reg == "dscratch") \
-               && $rgroup != "all" && $rgroup != "x_all"} {
-           gdb_assert {$count == 0} \
-               "register $reg not seen in reggroup $rgroup"
-       } else {
-           gdb_assert {$count == 1} \
-               "register $reg seen once in reggroup $rgroup"
-       }
+
+       set expected_count [ get_expected_result $reg $rgroup ]
+       gdb_assert {$count == $expected_count} \
+           "register $reg seen in reggroup $rgroup $expected_count times"
     }
     array unset reg_counts
 }