1 /*
  2  * Copyright (c) 2018, 2020, Red Hat, Inc. All rights reserved.
  3  * Copyright (c) 2021, 2023, Oracle and/or its affiliates. All rights reserved.
  4  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
  5  *
  6  * This code is free software; you can redistribute it and/or modify it
  7  * under the terms of the GNU General Public License version 2 only, as
  8  * published by the Free Software Foundation.
  9  *
 10  * This code is distributed in the hope that it will be useful, but WITHOUT
 11  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 12  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 13  * version 2 for more details (a copy is included in the LICENSE file that
 14  * accompanied this code).
 15  *
 16  * You should have received a copy of the GNU General Public License version
 17  * 2 along with this work; if not, write to the Free Software Foundation,
 18  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 19  *
 20  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 21  * or visit www.oracle.com if you need additional information or have any
 22  * questions.
 23  */
 24 
 25 /*
 26  * common code to run and validate tests of code generation for
 27  * volatile ops on AArch64
 28  *
 29  * incoming args are <testclass> <testtype>
 30  *
 31  * where <testclass> in {TestVolatileLoad,
 32  *                       TestVolatileStore,
 33  *                       TestUnsafeVolatileLoad,
 34  *                       TestUnsafeVolatileStore,
 35  *                       TestUnsafeVolatileCAS,
 36  *                       TestUnsafeVolatileWeakCAS,
 37  *                       TestUnsafeVolatileCAE,
 38  *                       TestUnsafeVolatileGAS}
 39  * and <testtype> in {G1,
 40  *                    Serial,
 41  *                    Parallel,
 42  *                    Shenandoah,
 43  *                    ShenandoahIU}
 44  */
 45 
 46 
 47 package compiler.c2.aarch64;
 48 
 49 import java.util.List;
 50 import java.util.ListIterator;
 51 import java.util.Iterator;
 52 import java.util.regex.Pattern;
 53 import java.io.*;
 54 
 55 import jdk.test.lib.Asserts;
 56 import jdk.test.lib.compiler.InMemoryJavaCompiler;
 57 import jdk.test.lib.process.OutputAnalyzer;
 58 import jdk.test.lib.process.ProcessTools;
 59 import jdk.test.whitebox.WhiteBox;
 60 
 61 // runner class that spawns a new JVM to exercises a combination of
 62 // volatile MemOp and GC. The ops are compiled with the dmb -->
 63 // ldar/stlr transforms either enabled or disabled. this runner parses
 64 // the PrintOptoAssembly output checking that the generated code is
 65 // correct.
 66 
 67 public class TestVolatiles {
 68     public void runtest(String classname, String testType) throws Throwable {
 69         // n.b. clients omit the package name for the class
 70         String fullclassname = "compiler.c2.aarch64." + classname;
 71         // build up a command line for the spawned JVM
 72         String[] procArgs;
 73         int argcount;
 74         // add one or two extra arguments according to test type
 75         // i.e. GC type plus GC conifg
 76         switch(testType) {
 77         case "G1":
 78             argcount = 9;
 79             procArgs = new String[argcount];
 80             procArgs[argcount - 2] = "-XX:+UseG1GC";
 81             break;
 82         case "Parallel":
 83             argcount = 9;
 84             procArgs = new String[argcount];
 85             procArgs[argcount - 2] = "-XX:+UseParallelGC";
 86             break;
 87         case "Serial":
 88             argcount = 9;
 89             procArgs = new String[argcount];
 90             procArgs[argcount - 2] = "-XX:+UseSerialGC";
 91             break;
 92         case "Shenandoah":
 93             argcount = 10;
 94             procArgs = new String[argcount];
 95             procArgs[argcount - 3] = "-XX:+UnlockExperimentalVMOptions";
 96             procArgs[argcount - 2] = "-XX:+UseShenandoahGC";
 97             break;
 98         case "ShenandoahIU":
 99             argcount = 11;
100             procArgs = new String[argcount];
101             procArgs[argcount - 4] = "-XX:+UnlockExperimentalVMOptions";
102             procArgs[argcount - 3] = "-XX:+UseShenandoahGC";
103             procArgs[argcount - 2] = "-XX:ShenandoahGCMode=iu";
104             break;
105         default:
106             throw new RuntimeException("unexpected test type " + testType);
107         }
108 
109         // fill in arguments common to all cases
110 
111         // the first round of test enables transform of barriers to
112         // use acquiring loads and releasing stores by setting arg
113         // zero appropriately. this arg is reset in the second run to
114         // disable the transform.
115 
116         procArgs[0] = "-XX:+UseCompressedOops";
117         procArgs[1] = "-XX:-BackgroundCompilation";
118         procArgs[2] = "-XX:-TieredCompilation";
119         procArgs[3] = "-XX:+PrintOptoAssembly";
120         procArgs[4] = "-XX:CompileCommand=compileonly," + fullclassname + "::" + "test*";
121         procArgs[5] = "--add-exports";
122         procArgs[6] = "java.base/jdk.internal.misc=ALL-UNNAMED";
123         procArgs[argcount - 1] = fullclassname;
124 
125         runtest(classname, testType, true, procArgs);
126 
127         if (!classname.equals("TestUnsafeVolatileGAA")) {
128             procArgs[0] = "-XX:-UseCompressedOops";
129             runtest(classname, testType, false, procArgs);
130         }
131     }
132 
133 
134     public void runtest(String classname, String testType, boolean useCompressedOops, String[] procArgs) throws Throwable {
135         ProcessBuilder pb = ProcessTools.createLimitedTestJavaProcessBuilder(procArgs);
136         OutputAnalyzer output = new OutputAnalyzer(pb.start());
137 
138         output.stderrShouldBeEmptyIgnoreVMWarnings();
139         output.stdoutShouldNotBeEmpty();
140         output.shouldHaveExitValue(0);
141 
142         // check the output for the correct asm sequence as
143         // appropriate to test class, test type and whether transform
144         // was applied
145 
146         checkoutput(output, classname, testType, useCompressedOops);
147     }
148 
149     // skip through output returning a line containing the desireed
150     // substring or null
151     private String skipTo(Iterator<String> iter, String substring)
152     {
153         while (iter.hasNext()) {
154             String nextLine = iter.next();
155             if (nextLine.matches(".*" + substring + ".*")) {
156                 return nextLine;
157             }
158         }
159         return null;
160     }
161 
162     // locate the start of compiler output for the desired method and
163     // then check that each expected instruction occurs in the output
164     // in the order supplied. throw an excpetion if not found.
165     // n.b. the spawned JVM's output is included in the exception
166     // message to make it easeir to identify what is missing.
167 
168     private boolean checkCompile(Iterator<String> iter, String methodname, String[] expected, OutputAnalyzer output, boolean do_throw)
169     {
170         // trace call to allow eyeball check of what we are checking against
171         System.out.println("checkCompile(" + methodname + ",");
172         String sepr = "  { ";
173         for (String s : expected) {
174             System.out.print(sepr);
175             System.out.print(s);
176             sepr = ",\n    ";
177         }
178         System.out.println(" })");
179 
180         // look for the start of an opto assembly print block
181         String match = skipTo(iter, Pattern.quote("{method}"));
182         if (match == null) {
183             if (do_throw) {
184                 throw new RuntimeException("Missing compiler output for " + methodname + "!\n\n" + output.getOutput());
185             }
186             return false;
187         }
188         // check the compiled method name is right
189         match = skipTo(iter, Pattern.quote("- name:"));
190         if (match == null) {
191             if (do_throw) {
192                 throw new RuntimeException("Missing compiled method name!\n\n" + output.getOutput());
193             }
194             return false;
195         }
196         if (!match.contains(methodname)) {
197             if (do_throw) {
198                 throw new RuntimeException("Wrong method " + match + "!\n  -- expecting " + methodname + "\n\n" + output.getOutput());
199             }
200             return false;
201         }
202         // make sure we can match each expected term in order
203         for (String s : expected) {
204             match = skipTo(iter, s);
205             if (match == null) {
206                 if (do_throw) {
207                     throw new RuntimeException("Missing expected output " + s + "!\n\n" + output.getOutput());
208                 }
209                 return false;
210             }
211         }
212         return true;
213     }
214 
215     // check for expected asm output from a volatile load
216 
217     private void checkload(OutputAnalyzer output, String testType, boolean useCompressedOops) throws Throwable
218     {
219         Iterator<String> iter = output.asLines().listIterator();
220 
221         // we shoud see this same sequence for normal or unsafe volatile load
222         // for both int and Object fields
223 
224         String[] matches;
225         matches = new String[] {
226             "ldarw",
227             "membar_acquire \\(elided\\)",
228             "ret"
229         };
230         checkCompile(iter, "testInt", matches, output, true);
231 
232         matches = new String[] {
233             useCompressedOops ? "ldarw?" : "ldar",
234             "membar_acquire \\(elided\\)",
235             "ret"
236         };
237         checkCompile(iter, "testObj", matches, output, true);
238 
239     }
240 
241     // check for expected asm output from a volatile store
242 
243     private void checkstore(OutputAnalyzer output, String testType, boolean useCompressedOops) throws Throwable
244     {
245         Iterator<String> iter = output.asLines().listIterator();
246 
247         String[] matches;
248 
249         // non object stores are straightforward
250         // this is the sequence of instructions for all cases
251         matches = new String[] {
252             "membar_release \\(elided\\)",
253             "stlrw",
254             "membar_volatile \\(elided\\)",
255             "ret"
256         };
257         checkCompile(iter, "testInt", matches, output, true);
258 
259         // object stores will be as above except for when the GC
260         // introduces barriers for card marking
261         switch (testType) {
262         default:
263             // this is the basic sequence of instructions
264             matches = new String[] {
265                 "membar_release \\(elided\\)",
266                 useCompressedOops ? "stlrw?" : "stlr",
267                 "membar_volatile \\(elided\\)",
268                 "ret"
269             };
270             break;
271         case "G1":
272             // a card mark volatile barrier should be generated
273             // before the card mark strb
274             //
275             // following the fix for 8225776 the G1 barrier is now
276             // scheduled out of line after the membar volatile and
277             // and subsequent return
278             matches = new String[] {
279                 "membar_release \\(elided\\)",
280                 useCompressedOops ? "stlrw?" : "stlr",
281                 "membar_volatile \\(elided\\)",
282                 "ret",
283                 "membar_volatile",
284                 "dmb ish",
285                 "strb"
286             };
287             break;
288         case "Shenandoah":
289         case "ShenandoahIU":
290              // Shenandoah generates normal object graphs for
291              // volatile stores
292             matches = new String[] {
293                 "membar_release \\(elided\\)",
294                 useCompressedOops ? "stlrw?" : "stlr",
295                 "membar_volatile \\(elided\\)",
296                 "ret"
297             };
298             break;
299         }
300 
301         checkCompile(iter, "testObj", matches, output, true);
302     }
303 
304     // check for expected asm output from a volatile cas
305 
306     private void checkcas(OutputAnalyzer output, String testType, boolean useCompressedOops) throws Throwable
307     {
308         Iterator<String> iter = output.asLines().listIterator();
309 
310         String[] matches;
311         String[][] tests = {
312             { "testInt", "cmpxchgw" },
313             { "testLong", "cmpxchg" },
314             { "testByte", "cmpxchgb" },
315             { "testShort", "cmpxchgs" },
316         };
317 
318         for (String[] test : tests) {
319             // non object stores are straightforward
320             // this is the sequence of instructions for all cases
321             matches = new String[] {
322                 "membar_release \\(elided\\)",
323                 test[1] + "_acq",
324                 "membar_acquire \\(elided\\)",
325                 "ret"
326             };
327             checkCompile(iter, test[0], matches, output, true);
328         }
329 
330         // object stores will be as above except for when the GC
331         // introduces barriers for card marking
332         switch (testType) {
333         default:
334             // this is the basic sequence of instructions
335             matches = new String[] {
336                 "membar_release \\(elided\\)",
337                 useCompressedOops ? "cmpxchgw?_acq" : "cmpxchg_acq",
338                 "strb",
339                 "membar_acquire \\(elided\\)",
340                 "ret"
341             };
342             break;
343         case "G1":
344             // a card mark volatile barrier should be generated
345             // before the card mark strb
346             //
347             // following the fix for 8225776 the G1 barrier is now
348             // scheduled out of line after the membar acquire and
349             // and subsequent return
350             matches = new String[] {
351                 "membar_release \\(elided\\)",
352                 useCompressedOops ? "cmpxchgw?_acq" : "cmpxchg_acq",
353                 "membar_acquire \\(elided\\)",
354                 "ret",
355                 "membar_volatile",
356                 "dmb ish",
357                 "strb"
358             };
359             break;
360         case "Shenandoah":
361         case "ShenandoahIU":
362             // For volatile CAS, Shenanodoah generates normal
363             // graphs with a shenandoah-specific cmpxchg
364             matches = new String[] {
365                 "membar_release \\(elided\\)",
366                 useCompressedOops ? "cmpxchgw?_acq_shenandoah" : "cmpxchg_acq_shenandoah",
367                 "membar_acquire \\(elided\\)",
368                 "ret"
369             };
370             break;
371         }
372         checkCompile(iter, "testObj", matches, output, true);
373     }
374 
375     private void checkcae(OutputAnalyzer output, String testType, boolean useCompressedOops) throws Throwable
376     {
377         ListIterator<String> iter = output.asLines().listIterator();
378 
379         String[] matches;
380         String[][] tests = {
381             { "testInt", "cmpxchgw" },
382             { "testLong", "cmpxchg" },
383             { "testByte", "cmpxchgb" },
384             { "testShort", "cmpxchgs" },
385         };
386 
387         for (String[] test : tests) {
388             // non object stores are straightforward
389             // this is the sequence of instructions for all cases
390             matches = new String[] {
391                 "membar_release \\(elided\\)",
392                 test[1] + "_acq",
393                 "membar_acquire \\(elided\\)",
394                 "ret"
395             };
396             checkCompile(iter, test[0], matches, output, true);
397         }
398 
399         // object stores will be as above except for when the GC
400         // introduces barriers for card marking
401         switch (testType) {
402         default:
403             // this is the basic sequence of instructions
404             matches = new String[] {
405                 "membar_release \\(elided\\)",
406                 "strb",
407                 useCompressedOops ? "cmpxchgw?_acq" : "cmpxchg_acq",
408                 "membar_acquire \\(elided\\)",
409                 "ret"
410             };
411 
412             // card marking store may be scheduled before or after
413             // the cmpxchg so try both sequences.
414             int idx = iter.nextIndex();
415             if (!checkCompile(iter, "testObj", matches, output, false)) {
416                 iter = output.asLines().listIterator(idx);
417 
418                 matches = new String[] {
419                     "membar_release \\(elided\\)",
420                     useCompressedOops ? "cmpxchgw?_acq" : "cmpxchg_acq",
421                     "strb",
422                     "membar_acquire \\(elided\\)",
423                     "ret"
424                 };
425 
426                 checkCompile(iter, "testObj", matches, output, true);
427             }
428             return;
429 
430         case "G1":
431             // a card mark volatile barrier should be generated
432             // before the card mark strb
433             //
434             // following the fix for 8225776 the G1 barrier is now
435             // scheduled out of line after the membar acquire and
436             // and subsequent return
437             matches = new String[] {
438                 "membar_release \\(elided\\)",
439                 useCompressedOops ? "cmpxchgw?_acq" : "cmpxchg_acq",
440                 "membar_acquire \\(elided\\)",
441                 "ret",
442                 "membar_volatile",
443                 "dmb ish",
444                 "strb"
445             };
446             break;
447         case "Shenandoah":
448         case "ShenandoahIU":
449             // For volatile CAS, Shenanodoah generates normal
450             // graphs with a shenandoah-specific cmpxchg
451             matches = new String[] {
452                 "membar_release \\(elided\\)",
453                 useCompressedOops ? "cmpxchgw?_acq_shenandoah" : "cmpxchg_acq_shenandoah",
454                 "membar_acquire \\(elided\\)",
455                 "ret"
456             };
457             break;
458         }
459         checkCompile(iter, "testObj", matches, output, true);
460     }
461 
462     private void checkgas(OutputAnalyzer output, String testType, boolean useCompressedOops) throws Throwable
463     {
464         Iterator<String> iter = output.asLines().listIterator();
465 
466         String[] matches;
467         String[][] tests = {
468             { "testInt", "atomic_xchgw" },
469             { "testLong", "atomic_xchg" },
470         };
471 
472         for (String[] test : tests) {
473             // non object stores are straightforward
474             // this is the sequence of instructions for all cases
475             matches = new String[] {
476                 "membar_release \\(elided\\)",
477                 test[1] + "_acq",
478                 "membar_acquire \\(elided\\)",
479                 "ret"
480             };
481             checkCompile(iter, test[0], matches, output, true);
482         }
483 
484         // object stores will be as above except for when the GC
485         // introduces barriers for card marking
486         switch (testType) {
487         default:
488             // this is the basic sequence of instructions
489             matches = new String[] {
490                 "membar_release \\(elided\\)",
491                 useCompressedOops ? "atomic_xchgw?_acq" : "atomic_xchg_acq",
492                 "strb",
493                 "membar_acquire \\(elided\\)",
494                 "ret"
495             };
496             break;
497         case "G1":
498             // a card mark volatile barrier should be generated
499             // before the card mark strb
500             //
501             // following the fix for 8225776 the G1 barrier is now
502             // scheduled out of line after the membar acquire and
503             // and subsequent return
504             matches = new String[] {
505                 "membar_release \\(elided\\)",
506                 useCompressedOops ? "atomic_xchgw?_acq" : "atomic_xchg_acq",
507                 "membar_acquire \\(elided\\)",
508                 "ret",
509                 "membar_volatile",
510                 "dmb ish",
511                 "strb"
512             };
513             break;
514         case "Shenandoah":
515         case "ShenandoahIU":
516             matches = new String[] {
517                 "membar_release \\(elided\\)",
518                 useCompressedOops ? "atomic_xchgw?_acq" : "atomic_xchg_acq",
519                 "membar_acquire \\(elided\\)",
520                 "ret"
521             };
522             break;
523         }
524 
525         checkCompile(iter, "testObj", matches, output, true);
526     }
527 
528     private void checkgaa(OutputAnalyzer output, String testType) throws Throwable
529     {
530         Iterator<String> iter = output.asLines().listIterator();
531 
532         String[] matches;
533         String[][] tests = {
534             { "testInt", "get_and_addI" },
535             { "testLong", "get_and_addL" },
536         };
537 
538         for (String[] test : tests) {
539             // non object stores are straightforward
540             // this is the sequence of instructions for all cases
541             matches = new String[] {
542                 "membar_release \\(elided\\)",
543                 test[1] + "_acq",
544                 "membar_acquire \\(elided\\)",
545                 "ret"
546             };
547             checkCompile(iter, test[0], matches, output, true);
548         }
549 
550     }
551 
552     // perform a check appropriate to the classname
553 
554     private void checkoutput(OutputAnalyzer output, String classname, String testType, boolean useCompressedOops) throws Throwable
555     {
556         // trace call to allow eyeball check of what is being checked
557         System.out.println("checkoutput(" +
558                            classname + ", " +
559                            testType + ")\n" +
560                            output.getOutput());
561 
562         switch (classname) {
563         case "TestVolatileLoad":
564             checkload(output, testType, useCompressedOops);
565             break;
566         case "TestVolatileStore":
567             checkstore(output, testType, useCompressedOops);
568             break;
569         case "TestUnsafeVolatileLoad":
570             checkload(output, testType, useCompressedOops);
571             break;
572         case "TestUnsafeVolatileStore":
573             checkstore(output, testType, useCompressedOops);
574             break;
575         case "TestUnsafeVolatileCAS":
576         case "TestUnsafeVolatileWeakCAS":
577             checkcas(output, testType, useCompressedOops);
578             break;
579         case "TestUnsafeVolatileCAE":
580             checkcae(output, testType, useCompressedOops);
581             break;
582         case "TestUnsafeVolatileGAS":
583             checkgas(output, testType, useCompressedOops);
584             break;
585         case "TestUnsafeVolatileGAA":
586             checkgaa(output, testType);
587             break;
588         }
589     }
590 }