summaryrefslogtreecommitdiff
path: root/src/sim
diff options
context:
space:
mode:
authorMoyang Wang <mw828@cornell.edu>2018-04-02 16:23:02 -0400
committerTuan Ta <qtt2@cornell.edu>2019-02-08 15:25:30 +0000
commit758b62cfb2ede5fa2187e4cab899691505179d43 (patch)
treedeaaf97ea7b97c061f36e10d08463ae78498e864 /src/sim
parent42b063ad30324f1459915070226e820aabd54336 (diff)
downloadgem5-758b62cfb2ede5fa2187e4cab899691505179d43.tar.xz
sim, kern: support FUTEX_CMP_REQUEUE
This patch supports FUTEX_CMP_REQUEUE operation. Below is its description from Linux man page: futex syscall: int futex(int *uaddr, int futex_op, int val, const struct timespec *timeout, int *uaddr2, int val3); This operation first checks whether the location uaddr still contains the value val3. If not, the operation fails with the error EAGAIN. Otherwise, the operation wakes up a maximum of val waiters that are waiting on the futex at uaddr. If there are more than val waiters, then the remaining waiters are removed from the wait queue of the source futex at uaddr and added to the wait queue of the target futex at uaddr2. The val2 argument specifies an upper limit on the number of waiters that are requeued to the futex at uaddr2. Reference: http://man7.org/linux/man-pages/man2/futex.2.html Change-Id: I6d2ebd19a935b656d19d8342f7ab450c0d2031f4 Reviewed-on: https://gem5-review.googlesource.com/c/9629 Reviewed-by: Brandon Potter <Brandon.Potter@amd.com> Maintainer: Brandon Potter <Brandon.Potter@amd.com>
Diffstat (limited to 'src/sim')
-rw-r--r--src/sim/futex_map.hh54
-rw-r--r--src/sim/syscall_emul.hh15
2 files changed, 68 insertions, 1 deletions
diff --git a/src/sim/futex_map.hh b/src/sim/futex_map.hh
index 6f1f7a2a1..3d34109be 100644
--- a/src/sim/futex_map.hh
+++ b/src/sim/futex_map.hh
@@ -221,6 +221,60 @@ class FutexMap : public std::unordered_map<FutexKey, WaiterList>
return woken_up;
}
+
+ /**
+ * This operation wakes a given number (val) of waiters. If there are
+ * more threads waiting than woken, they are removed from the wait
+ * queue of the futex pointed to by addr1 and added to the wait queue
+ * of the futex pointed to by addr2. The number of waiter moved is
+ * capped by count2 (misused timeout parameter).
+ *
+ * The return value is the number of waiters that are woken or
+ * requeued.
+ */
+ int
+ requeue(Addr addr1, uint64_t tgid, int count, int count2, Addr addr2)
+ {
+ FutexKey key1(addr1, tgid);
+ auto it1 = find(key1);
+
+ if (it1 == end())
+ return 0;
+
+ int woken_up = 0;
+ auto &waiterList1 = it1->second;
+
+ while (!waiterList1.empty() && woken_up < count) {
+ waiterList1.front().tc->activate();
+ waiterList1.pop_front();
+ woken_up++;
+ }
+
+ WaiterList tmpList;
+ int requeued = 0;
+
+ while (!waiterList1.empty() && requeued < count2) {
+ auto w = waiterList1.front();
+ waiterList1.pop_front();
+ tmpList.push_back(w);
+ requeued++;
+ }
+
+ FutexKey key2(addr2, tgid);
+ auto it2 = find(key2);
+
+ if (it2 == end() && requeued > 0) {
+ insert({key2, tmpList});
+ } else {
+ it2->second.insert(it2->second.end(),
+ tmpList.begin(), tmpList.end());
+ }
+
+ if (waiterList1.empty())
+ erase(it1);
+
+ return woken_up + requeued;
+ }
};
#endif // __FUTEX_MAP_HH__
diff --git a/src/sim/syscall_emul.hh b/src/sim/syscall_emul.hh
index 295598c52..0b7585cb4 100644
--- a/src/sim/syscall_emul.hh
+++ b/src/sim/syscall_emul.hh
@@ -441,8 +441,21 @@ futexFunc(SyscallDesc *desc, int callnum, Process *process,
return futex_map.wakeup(uaddr, process->tgid(), val);
} else if (OS::TGT_FUTEX_WAKE_BITSET == op) {
return futex_map.wakeup_bitset(uaddr, process->tgid(), val3);
- }
+ } else if (OS::TGT_FUTEX_REQUEUE == op ||
+ OS::TGT_FUTEX_CMP_REQUEUE == op) {
+ // Ensure futex system call accessed atomically.
+ BufferArg buf(uaddr, sizeof(int));
+ buf.copyIn(tc->getMemProxy());
+ int mem_val = *(int*)buf.bufferPtr();
+ /*
+ * For CMP_REQUEUE, the whole operation is only started only if
+ * val3 is still the value of the futex pointed to by uaddr.
+ */
+ if (OS::TGT_FUTEX_CMP_REQUEUE && val3 != mem_val)
+ return -OS::TGT_EWOULDBLOCK;
+ return futex_map.requeue(uaddr, process->tgid(), val, timeout, uaddr2);
+ }
warn("futex: op %d not implemented; ignoring.", op);
return -ENOSYS;
}