80 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			80 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			Java
		
	
	
	
| /*
 | |
|  * Copyright (C) 2018 The Android Open Source Project
 | |
|  *
 | |
|  * Licensed under the Apache License, Version 2.0 (the "License");
 | |
|  * you may not use this file except in compliance with the License.
 | |
|  * You may obtain a copy of the License at
 | |
|  *
 | |
|  *      http://www.apache.org/licenses/LICENSE-2.0
 | |
|  *
 | |
|  * Unless required by applicable law or agreed to in writing, software
 | |
|  * distributed under the License is distributed on an "AS IS" BASIS,
 | |
|  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
|  * See the License for the specific language governing permissions and
 | |
|  * limitations under the License.
 | |
|  */
 | |
| 
 | |
| import java.lang.reflect.Field;
 | |
| import java.lang.reflect.Method;
 | |
| 
 | |
| class NotLoaded {
 | |
|   public void foo() {}
 | |
| }
 | |
| 
 | |
| public class Main {
 | |
|   public static void main(String[] args) throws Exception {
 | |
|     runTest(null);
 | |
|   }
 | |
| 
 | |
|   // This method being synchronized means the SIGQUIT code in ART will call
 | |
|   // FindLocksAtDexPc (we check for the presence of try blocks and monitor-enter),
 | |
|   // which triggered a DCHECK of an invariant.
 | |
|   public static synchronized void runTest(Object m) throws Exception {
 | |
|     if (m != null) {
 | |
|       // We used to crash while trying to resolve NotLoaded and beint interrupted
 | |
|       // by the SIGQUIT.
 | |
|       if (m instanceof NotLoaded) {
 | |
|         ((NotLoaded)m).foo();
 | |
|       }
 | |
|     }
 | |
|     SigQuit.doKill();
 | |
|     // Sleep some time to get the kill while executing this method.
 | |
|     Thread.sleep(2);
 | |
|     System.out.println("Done");
 | |
|   }
 | |
| 
 | |
|   private final static class SigQuit {
 | |
|     private final static int sigquit;
 | |
|     private final static Method kill;
 | |
|     private final static int pid;
 | |
| 
 | |
|     static {
 | |
|       int pidTemp = -1;
 | |
|       int sigquitTemp = -1;
 | |
|       Method killTemp = null;
 | |
| 
 | |
|       try {
 | |
|         Class<?> osClass = Class.forName("android.system.Os");
 | |
|         Method getpid = osClass.getDeclaredMethod("getpid");
 | |
|         pidTemp = (Integer)getpid.invoke(null);
 | |
| 
 | |
|         Class<?> osConstants = Class.forName("android.system.OsConstants");
 | |
|         Field sigquitField = osConstants.getDeclaredField("SIGQUIT");
 | |
|         sigquitTemp = (Integer)sigquitField.get(null);
 | |
| 
 | |
|         killTemp = osClass.getDeclaredMethod("kill", int.class, int.class);
 | |
|       } catch (Exception e) {
 | |
|         throw new Error(e);
 | |
|       }
 | |
| 
 | |
|       pid = pidTemp;
 | |
|       sigquit = sigquitTemp;
 | |
|       kill = killTemp;
 | |
|     }
 | |
| 
 | |
|     public static void doKill() throws Exception {
 | |
|       kill.invoke(null, pid, sigquit);
 | |
|     }
 | |
|   }
 | |
| }
 |