Lines Matching +full:parent +full:- +full:child
1 // SPDX-License-Identifier: GPL-2.0+
9 #include "child.h"
29 #define pkeyshift(pkey) (PKEY_REG_BITS - ((pkey + 1) * AMR_BITS_PER_PKEY))
36 /* Information shared between the parent and the child. */
40 /* AMR value the parent expects to read from the child. */
43 /* AMR value the parent is expected to write to the child. */
46 /* AMR value that ptrace should refuse to write to the child. */
49 /* IAMR value the parent expects to read from the child. */
52 /* UAMOR value the parent expects to read from the child. */
56 * IAMR and UAMOR values that ptrace should refuse to write to the child
69 static int child(struct shared_info *info) in child() function
76 /* Wait until parent fills out the initial register values. */ in child()
77 ret = wait_parent(&info->child_sync); in child()
85 CHILD_FAIL_IF(pkey1 < 0, &info->child_sync); in child()
91 CHILD_FAIL_IF(pkey2 < 0, &info->child_sync); in child()
94 CHILD_FAIL_IF(pkey3 < 0, &info->child_sync); in child()
96 info->amr1 |= 3ul << pkeyshift(pkey1); in child()
97 info->amr2 |= 3ul << pkeyshift(pkey2); in child()
102 info->invalid_amr = info->amr2 | (~0x0UL & ~info->expected_uamor); in child()
108 info->expected_iamr |= 1ul << pkeyshift(pkey1); in child()
110 info->expected_iamr &= ~(1ul << pkeyshift(pkey1)); in child()
115 info->expected_iamr &= ~(1ul << pkeyshift(pkey2)); in child()
116 info->expected_iamr &= ~(1ul << pkeyshift(pkey3)); in child()
122 info->invalid_iamr = info->expected_iamr | (1ul << pkeyshift(pkey1) | 1ul << pkeyshift(pkey2)); in child()
123 info->invalid_uamor = info->expected_uamor & ~(0x3ul << pkeyshift(pkey1)); in child()
125 printf("%-30s AMR: %016lx pkey1: %d pkey2: %d pkey3: %d\n", in child()
126 user_write, info->amr1, pkey1, pkey2, pkey3); in child()
128 set_amr(info->amr1); in child()
130 /* Wait for parent to read our AMR value and write a new one. */ in child()
131 ret = prod_parent(&info->child_sync); in child()
132 CHILD_FAIL_IF(ret, &info->child_sync); in child()
134 ret = wait_parent(&info->child_sync); in child()
140 printf("%-30s AMR: %016lx\n", user_read, reg); in child()
142 CHILD_FAIL_IF(reg != info->amr2, &info->child_sync); in child()
145 * Wait for parent to try to write an invalid AMR value. in child()
147 ret = prod_parent(&info->child_sync); in child()
148 CHILD_FAIL_IF(ret, &info->child_sync); in child()
150 ret = wait_parent(&info->child_sync); in child()
156 printf("%-30s AMR: %016lx\n", user_read, reg); in child()
158 CHILD_FAIL_IF(reg != info->amr2, &info->child_sync); in child()
161 * Wait for parent to try to write an IAMR and a UAMOR value. We can't in child()
164 ret = prod_parent(&info->child_sync); in child()
165 CHILD_FAIL_IF(ret, &info->child_sync); in child()
167 ret = wait_parent(&info->child_sync); in child()
173 printf("%-30s AMR: %016lx\n", user_read, reg); in child()
175 CHILD_FAIL_IF(reg != info->amr2, &info->child_sync); in child()
177 /* Now let parent now that we are finished. */ in child()
179 ret = prod_parent(&info->child_sync); in child()
180 CHILD_FAIL_IF(ret, &info->child_sync); in child()
185 static int parent(struct shared_info *info, pid_t pid) in parent() function
192 * to the child. in parent()
195 PARENT_SKIP_IF_UNSUPPORTED(ret, &info->child_sync); in parent()
196 PARENT_FAIL_IF(ret, &info->child_sync); in parent()
198 info->amr1 = info->amr2 = regs[0]; in parent()
199 info->expected_iamr = regs[1]; in parent()
200 info->expected_uamor = regs[2]; in parent()
202 /* Wake up child so that it can set itself up. */ in parent()
203 ret = prod_child(&info->child_sync); in parent()
204 PARENT_FAIL_IF(ret, &info->child_sync); in parent()
206 ret = wait_child(&info->child_sync); in parent()
210 /* Verify that we can read the pkey registers from the child. */ in parent()
212 PARENT_FAIL_IF(ret, &info->child_sync); in parent()
214 printf("%-30s AMR: %016lx IAMR: %016lx UAMOR: %016lx\n", in parent()
217 PARENT_FAIL_IF(regs[0] != info->amr1, &info->child_sync); in parent()
218 PARENT_FAIL_IF(regs[1] != info->expected_iamr, &info->child_sync); in parent()
219 PARENT_FAIL_IF(regs[2] != info->expected_uamor, &info->child_sync); in parent()
221 /* Write valid AMR value in child. */ in parent()
222 ret = ptrace_write_regs(pid, NT_PPC_PKEY, &info->amr2, 1); in parent()
223 PARENT_FAIL_IF(ret, &info->child_sync); in parent()
225 printf("%-30s AMR: %016lx\n", ptrace_write_running, info->amr2); in parent()
227 /* Wake up child so that it can verify it changed. */ in parent()
228 ret = prod_child(&info->child_sync); in parent()
229 PARENT_FAIL_IF(ret, &info->child_sync); in parent()
231 ret = wait_child(&info->child_sync); in parent()
235 /* Write invalid AMR value in child. */ in parent()
236 ret = ptrace_write_regs(pid, NT_PPC_PKEY, &info->invalid_amr, 1); in parent()
237 PARENT_FAIL_IF(ret, &info->child_sync); in parent()
239 printf("%-30s AMR: %016lx\n", ptrace_write_running, info->invalid_amr); in parent()
241 /* Wake up child so that it can verify it didn't change. */ in parent()
242 ret = prod_child(&info->child_sync); in parent()
243 PARENT_FAIL_IF(ret, &info->child_sync); in parent()
245 ret = wait_child(&info->child_sync); in parent()
250 regs[0] = info->amr1; in parent()
251 regs[1] = info->invalid_iamr; in parent()
253 PARENT_FAIL_IF(!ret, &info->child_sync); in parent()
255 printf("%-30s AMR: %016lx IAMR: %016lx\n", in parent()
259 regs[2] = info->invalid_uamor; in parent()
261 PARENT_FAIL_IF(!ret, &info->child_sync); in parent()
263 printf("%-30s AMR: %016lx IAMR: %016lx UAMOR: %016lx\n", in parent()
268 PARENT_FAIL_IF(ret, &info->child_sync); in parent()
270 printf("%-30s AMR: %016lx IAMR: %016lx UAMOR: %016lx\n", in parent()
273 PARENT_FAIL_IF(regs[0] != info->amr2, &info->child_sync); in parent()
274 PARENT_FAIL_IF(regs[1] != info->expected_iamr, &info->child_sync); in parent()
275 PARENT_FAIL_IF(regs[2] != info->expected_uamor, &info->child_sync); in parent()
277 /* Wake up child so that it can verify AMR didn't change and wrap up. */ in parent()
278 ret = prod_child(&info->child_sync); in parent()
279 PARENT_FAIL_IF(ret, &info->child_sync); in parent()
283 printf("Child's exit status not captured\n"); in parent()
286 printf("Child exited abnormally\n"); in parent()
304 ret = init_child_sync(&info->child_sync); in ptrace_pkey()
313 ret = child(info); in ptrace_pkey()
315 ret = parent(info, pid); in ptrace_pkey()
320 destroy_child_sync(&info->child_sync); in ptrace_pkey()