data:image/s3,"s3://crabby-images/c4b42/c4b424e229f4e63283f9ab8a035f44e27671a63b" alt="Changing Java's Semantics for Handling Null Pointer"
19th International Symposium on Software Reliability Engineering Changing Java’s Semantics for Handling Null Pointer Exceptions Kinga Dobolyi∗ Westley Weimer∗ University of Virginia University of Virginia [email protected] [email protected] Abstract prone, especially when pointer values are created by exter- nal components or are part of a chain of object references. We envision a world where no exceptions are raised; in- We analyze programs to locate possible null pointer deref- stead, language semantics are changed so that operations erences and then insert null checks and error handling code. are total functions. Either an operation executes normally The error-handling code is specified at compile-time via or tailored recovery code is applied where exceptions would composable, context-sensitive recovery policies. Generated have been raised. As an initial step and evaluation of this handling code might, for example, create a default object of idea, we propose to transform programs so that null pointer an appropriate type to replace the null value, skip instruc- dereferences are handled automatically without a large run- tions, perform logging, restore invariants, or some combi- time overhead. We increase robustness by replacing code nation of the above. This approach is especially desirable that raises null pointer exceptions with error-handling code, in web services or dynamic web content, where users inter- allowing the program to continue execution. Our technique pret the final results with respect to an acceptability enve- first finds potential null pointer dereferences and then au- lope [23] and high availability is of paramount importance. tomatically transforms programs to insert null checks and Because program behavior is preserved when no null point- error-handling code. These transformations are guided ers are dereferenced, our approach can be applied to any by composable, context-sensitive recovery policies. Error- Java program. Instead of raising null pointer exceptions, we handling code may, for example, create default objects of change Java’s semantics for pointer dereferences to a total the appropriate types, or restore data structure invariants. mapping for all possible pointer values. Rather than hav- If no null pointers would be dereferenced, the transformed ing non-exceptional behavior defined only for valid pointer program behaves just as the original. dereferences, we generate recovery code for the null values We applied our transformation in experiments involving as well. We aim to transform programs so that null pointer multiple benchmarks, the Java Standard Library, and exter- exceptions are avoided and programs can continue execut- nally reported null pointer exceptions. Our technique was ing without incurring a high run-time cost in space or speed. able to handle the reported exceptions and allow the pro- Null pointer exceptions, while conceptually simple, re- grams to continue to do useful work, with an average execu- main prevalent in practice. Null pointer dereferences are tion time overhead of less than 1% and an average bytecode frequent [31], and have been reported as “a very serious space overhead of 22%. threat to the safety of programs” and are the most common error in Java programs [7]. Many classes of null pointer ex- 1. Introduction ceptions can be found automatically by static analyses [15]. Addressing such risks with fault-tolerance techniques is This paper introduces APPEND, an automated approach a promising avenue. For example, techniques that mask to preventing and handling null pointer exceptions in Java memory errors have successfully eliminated security vul- programs. Removing null pointer exceptions is an impor- nerabilities in servers [25]. tant first step on the road to dependable total functions. Some programming idioms make static null pointer anal- Checking for null pointers manually is tedious and error- yses unattractive. For example, many programs simplify database interaction by creating and populating objects with ∗This research was supported in part by National Science Foun- dation Grants CNS 0627523 and CNS 0716478, Air Force Office field values based on columns in database tables (e.g., [2]). of Scientific Research grant FA9550-07-1-0532, and NASA grant The validity or nullity of a reference to such an object de- NAS1-02117, as well as gifts from Microsoft Research. The infor- pends on what is stored in the database at run-time. Conser- mation presented here does not necessarily reflect their positions vative static analyses typically flag all such uses as potential or policies and no official endorsement should be inferred. null dereferences, but some reports may be viewed as spuri- 1071-9458/08 $25.00 © 2008 IEEE 47 DOI 10.1109/ISSRE.2008.59 Authorized licensed use limited to: University of Virginia Libraries. Downloaded on November 5, 2009 at 11:00 from IEEE Xplore. Restrictions apply. ous false positives if there are external invariants requiring serting null pointer checks is tedious and error-prone. Null the presence of certain objects. In addition, not all defect pointers can arise from program defects or violated as- reports from static analysis tools are addressed [33]. Pro- sumptions, but are perhaps more insidious when they result grams ship with known bugs [18], and resources may not from external sources or components. For example, many be available to fix null pointer errors. database APIs that convert table entries into objects for ease We propose a program transformation that automatically of programmer manipulation may return null objects if the inserts null checks and error handling code. No program requested entity is not in the database. Runtime dependency annotations are required, and developers need not wade on external systems (e.g., databases) can significantly re- through defect reports. Programs are modified according duce the effectiveness of testing in finding potential null to composable recovery policies. Recovery policies are ex- pointer dereferences [21, 28]. ecuted at compile-time and, depending on the context, re- 1 Person prs = database.getPerson(personID); covery code is inserted that is then executed at run-time if 2 println("Name: " + prs.getName()); the null checks return true. Recovery policies are concep- 3 println("Zipcode: " + prs.getAddr().getZip()); tually related to theorem prover tactics and tacticals or to certain classes of aspect-oriented programming. If no null In the example above, if the requested person is not in values are dereferenced at run-time, the transformed pro- the database or if the database has been corrupted, a null gram behaves just as the original program. If the original Person object will be returned. One standard defensive program would dereference a null value, the transformed approach is to guard statements with non-null predicates: program instead executes the policy-dictated error-handling 1 Person prs = database.getPerson(personID); code, such as creating a default value on the fly or not calcu- 2 if (prs != null) lating that expression. Previous research has suggested that 3 println("Name: " + prs.getName()); 4 if (prs != null && prs.getAddr() != null) programs might successfully continue even with discarded 5 println("Zipcode: " + prs.getAddr().getZip()); instructions (e.g., [24]); we present and measure a concrete, low-level, annotation-free version of such a system, and ex- This way, if a valid Person is returned, the informa- tend it to allow for user-specified actions. tion is printed out normally. If a null pointer is returned, We choose to work at the application rather than modi- whether as a valid part of the program API or as an invalid fying the existing null checking behavior of a Java Virtual record from the database, the null pointer dereference will Machine. This has the advantages of retaining portability be prevented. between different virtual machines and of conceptual sim- Note that a even when a valid Person object is re- plicity, and the disadvantages of requiring that all relevant turned, the Address object within the Person may be null, source code be processed in advance. and must also be explicitly checked. While this example Our transformation can be implemented directly atop is for an object from a database, any value that is deref- existing program transformation frameworks and dovetails erenced could be a null pointer, and should be checked to easily with standard development processes. It can be ap- avoid a null pointer exception (NPE). The number of NPEs plied to individual source or class files, entire programs, and encountered and the research devoted to preventing them is separate libraries, in any combination. The main contribu- a testament to the inconsistency of null pointer prevention tions of this paper are a presentation of our technique (Sec- in practice [15]. At the same time, manually placing checks tion 3, including our definition of soundness in Section 3.3), in the code is not only time-consuming and error-prone, but our notion of recovery policies (Section 4), and experimen- can also make the code more complex and difficult to read. tal evidence (Section 5) to support the claim that our ap- One real-world example of problematic handling of proach can handle null pointer exceptions in practice with NPEs comes from JTIDY, a tool for analyzing and trans- minimal execution time overhead and low code size over- forming HTML. This example is taken from a bug report head (Section 5.3). We begin with a motivating example. submitted by a user on a public mailing list.1 In the code below, the NPE occurs on line 36: 2 Motivating Examples 30 Doc xhtml = tidy.parseDOM(in, null); 31 // translate DOM for dom4j 32 DOMReader xmlReader = new DOMReader(); In this section we walk through the application of our 33 Document doc = xmlReader.read(xhtml); technique to a simple example and to a publicly-reported 34 defect. We illustrate the process taken by our automatic 35 Node table = doc.selectNode("/html/body"); transformation and highlight the difficulties in manually 36 System.err.println("table:" + table.asXML()); handling null pointer exceptions.
Details
-
File Typepdf
-
Upload Time-
-
Content LanguagesEnglish
-
Upload UserAnonymous/Not logged-in
-
File Pages10 Page
-
File Size-