From: Peizhao Ou Date: Fri, 10 Apr 2015 18:02:05 +0000 (-0700) Subject: changes X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=f7207fbf6086300565ee3f30c220fd37a891eb4e;p=cdsspec-compiler.git changes --- diff --git a/correctness-model/note.txt b/correctness-model/note.txt new file mode 100644 index 0000000..e69de29 diff --git a/correctness-model/writeup/.#implementation.tex.1.9 b/correctness-model/writeup/.#implementation.tex.1.9 new file mode 100644 index 0000000..a741b61 --- /dev/null +++ b/correctness-model/writeup/.#implementation.tex.1.9 @@ -0,0 +1,163 @@ +\mysection{Implementation}\label{sec:implementation} + +The goal of the \TOOL specification language is to +enable developers to write specifications against which concurrent data structures can be tested. +We can ensure a concurrent data structure is correct with +respect to an equivalent sequential data structure if for each +execution of the concurrent data structure, the equivalent sequential +history for the equivalent sequential data structure yields the same +results. + +The execution space for many concurrent data +structures is unbounded, meaning that in practice we cannot +verify correctness by checking individual executions. However, the +specifications can be used for unit testing. In practice, +many bugs can be exposed by model checking unit tests for +concurrent data structures. We have implemented the \TOOL checker as a +unit testing tool built upon the \cdschecker framework. The \TOOL checker +can exhaustively explore all behaviors for unit tests and provide +developers with diagnostic reports for executions that violate their +specification. + + +\mysubsection{Model Checker Framework} + +The \TOOL checker takes as input a complete execution trace from the +\cdschecker model checker. The \cdschecker framework operates at the abstraction level of +individual atomic operations and thus has neither information about method +calls nor which atomic operations serve as ordering points. Thus, we +extend the framework by adding {\it annotation} operations to +\cdschecker's traces, which record the necessary information to check the specifications +but have no effect on other operations. The \TOOL compiler +inserts code to generate the annotation actions to communicate to the \TOOL checker +plugin the critical events for checking the \TOOL specification. +These annotation actions then appear in \cdschecker's list of atomic +operations and make it convenient for \TOOL to construct +a sequential history from the execution because for any given method +call, its invocation event, its ordering points, and its response event are +sequentially ordered in the list. + +\mysubsection{Specification Compiler} + +The specification compiler translates an annotated +C/C++ program into an instrumented C/C++ program that will generate +execution traces containing the dynamic information needed to check +assertions and construct the sequential history. We next +describe the type of annotation actions that the \TOOL compiler +inserts into the instrumented program. + +\mypara{\bf Ordering Points:} Ordering points have a conditional guard expression +and a label. Potential ordering point annotation actions are +inserted immediately after the atomic operation that serves as +the potential ordering point. Ordering point check annotation actions are +inserted where they appear. + +\mypara{\bf Method Boundary:} To identify a method's boundaries, \TOOL +inserts {\it method\_begin} and {\it method\_end} annotations at the +beginning and end of methods. + +\mypara{\bf Sequential States and Methods:} Since checking occurs +after \cdschecker has completed an execution, the annotation +actions stores the values of any variables in the concurrent data +structure that the annotations reference. + +\mypara{\bf Side Effects and Assertions:} Side effects and assertions perform +their checks after an execution. The side effects and assertions are compiled +into methods and the equivalent sequential data structure's states are accessible to these methods. +With this encapsulation, +the \TOOL checker simply calls these functions to implement the side effects and assertions. + +\mypara{\bf Synchronization Checks:} The \TOOL checker performs +synchronization checks in two parts: compiling the rules and runtime data +collection. First, the \TOOL compiler numbers all methods and happens-before +checks uniquely. For example, the rule ``\code{Update->Read}'' can be represented as (1, 0, 2, 0), +which means instances of method 1 that satisfy condition 0 should {\it + synchronize with} instances method 2 that satisfy condition 0. In this case, + condition 0 means \code{true}. Then, +the \TOOL compiler generates code that communicates the +synchronization rules by passing an array of integer pairs. Runtime +collection is then implemented by performing the condition check at +each method invocation or response and then passing the method number +and happens before condition if the check is satisfied. + +\mysubsection{Dynamic Checking} + +At this point, we have an execution trace with the necessary +annotations to construct a sequential history and to check +the execution's correctness. However, before constructing the +sequential history, the \TOOL plugin first collects +the necessary information for each method call, which is the {\it + method\_begin} annotation, the ordering point annotations, the +happens-before checks, and the {\it method\_end} annotations. Since +all of the operations in the trace have thread +identifiers it is straightforward to extract the operations between the {\it + method\_begin} and {\it method\_end} annotations. + +\mypara{\bf Reorder Method Calls:} As discussed above, determining the +ordering of the ordering points is non-trivial under the +C/C++ memory model. This can be complicated by the fact +that the C/C++ memory model allows atomic loads to read from atomic +stores that appear later in the trace and that we do not maintain program order in our +correctness model. + +However, we can still leverage the reads-from relation and the modification-order +relation to order the ordering points that appear in typical data +structures. \TOOL uses the following rules to generate an ordering-point +ordering $\reltext{opo}$ relation on ordering points. Given two operations $X$ and $Y$ that are both ordering points: + +\begin{enumerate} +\item {\bf Reads-From:} $X \xrightarrow{rf} Y \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (write-write):} $X \xrightarrow{mo} Y \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (read-write):} $A \xrightarrow{mo} Y \ \wedge \ A \xrightarrow{rf} X \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (write-read):} $X \xrightarrow{mo} B \ \wedge \ B \xrightarrow{rf} Y \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (read-read):} $A \xrightarrow{mo} B \ \wedge \ A \xrightarrow{rf} X \ \wedge \ B \xrightarrow{rf} Y \Rightarrow X \xrightarrow{opo} Y$. + +\end{enumerate} + +\mypara{\bf Generating the Reordering:} +The \TOOL checker first builds an execution graph where the nodes are +method calls and the edges represent the +$\reltext{opo}$ ordering of the ordering points of the methods that +correspond to the source and destination nodes. Assuming the absence +of cycles in the execution graph, +the $\reltext{opo}$ ordering is used to generate the sequential history. +If two methods are not ordered by the +$\reltext{opo}$ order, we assume that they commute and select an +arbitrary ordering for the methods. The \TOOL checker +topologically sorts the graph to generate the equivalent sequential +execution. + +When \TOOL fails to order ordering points, the operations often commute. Thus, +if multiple histories satisfy the constraints of \reltext{opo}, we generally +pick a random one for most data structures. However, when those operations +do not commute, we provide developers with different options: (1) they can +add additional ordering points to allow \TOOL to order the two nodes or +(2) they run \TOOL in either of the following modes: (a) \textit{loosely +exhaustive} mode --- \TOOL explores +all possible histories and only requires some history to be correct or +(b) \textit{strictly exhaustive} mode --- \TOOL explores all possible histories and requires all histories to +be correct. With these options, developers are able to balance the +tradeoff between specifying more ordering points and the searching +space. For example, for a single producer single +consumer queue based on a linked-list (maintaing \code{head} and \code{tail} pointer), if we only +specify the update and load of the next field of a linked-list node as ordering points for +\code{enqueue} and \code{dequeue}, \TOOL will fail to order a dequeue +returning \code{NULL} and another enqueue because dequeue does not have the +ordering point of loading the next field. In this case, developers can either +add extra ordering points to order the dequeue before the enqueue or simply run it in the +\textit{loosely exhaustive} mode and that mode will produce at least one correct +history. + +\mypara{\bf Check Synchronization Properties:} Synchronization properties are +specified using the IDs and conditions of method calls, and we have that information +ready after \TOOL constructs the sequential history and checks the preconditions +and postconditions. For two specific method calls $c_1$ and $c_2$, we can ensure +$c_1$ synchronizes with $c_2$ by ensuring the annotation \code{c1\_begin} +happens-before the annotation \code{c2\_end} because any operations +sequenced-before \code{c1\_begin} should happen-before any operations +sequenced-after \code{c2\_end} according to the C/C++11 memory model. + diff --git a/correctness-model/writeup/.#introduction.tex.1.24 b/correctness-model/writeup/.#introduction.tex.1.24 new file mode 100644 index 0000000..02c4ff7 --- /dev/null +++ b/correctness-model/writeup/.#introduction.tex.1.24 @@ -0,0 +1,222 @@ +\mysection{Introduction}\label{sec:introduction} + +Concurrent data structure design can improve scalability by supporting +multiple simultaneous operations, reducing memory coherence traffic, and reducing the time taken by an +individual data structure operation. Researchers have developed many concurrent +data structure designs with these goals~\cite{rcu, +lockfreequeue}. Concurrent data structures often use sophisticated techniques +including low-level atomic instructions (e.g., compare and +swap), careful reasoning about the order of loads and +stores, and fine-grained locking. For example, while the standard Java hash +table implementation can limit scalability to a handful of cores, +more sophisticated concurrent hash tables can scale to many hundreds of +cores~\cite{javaConcurrentHashMap}. + +The C/C++ standard committee extended the C and C++ languages with +support for low-level atomic operations in the C/C++11 +standard~\cite{cpp11spec, c11spec, boehmpldi} to enable developers to +write portable implementations of concurrent data structures. To support the relaxations typically performed by compilers and processors, +the +C/C++ memory model provides weaker semantics than sequential +consistency~\cite{scmemorymodel} and as a result, correctly using +these operations is challenging. Developers must not only +reason about potential interleavings, but also about how the processor +and compiler might reorder memory operations. Even experts make +subtle errors when reasoning about such memory models. + +Researchers have developed tools for exploring the behavior of code +under the C/C++ model including \cdschecker~\cite{cdschecker}, {\sc +Cppmem}~\cite{c11popl}, and Relacy~\cite{relacy}. These tools explore +behaviors that are allowed under the C/C++ memory model. While these +tools can certainly be useful for exploring executions, they can be +challenging to use for testing as they don't provide support (other +than assertions) for specifying the behavior of data structures. +Using assertions can be challenging as different interleavings or +reorderings legitimately produce different behaviors, and it can be +very difficult to code assertions to check the output of a test case under for an +arbitrary (unknown) execution. + +This paper presents \TOOL, a specification language and specification +checking tool that is designed to be used in conjunction with model +checking tools. We have implemented it as a plugin for +the \cdschecker model checker. + +\mysubsection{Background on Linearizability} + +Researchers have developed several techniques for specifying +correctness properties of concurrent data structures written for strong memory +models. While these techniques cannot handle the behaviors typically +exhibited by relaxed data structure implementations, they provide insight into +intuitive approaches to specifying concurrent data structure behavior. + +One approach for specifying the correctness of concurrent data +structures is in terms of equivalent sequential executions of either +the concurrent data structure or a simplified sequential version. The +problem then becomes how do we map a concurrent execution to an equivalent +sequential execution? A common criterion is {\it + linearizability} --- linearizability simply states that a concurrent +operation can be viewed as taking effect at some time between its +invocation and its return (or response)~\cite{linearizableref}. + +An {\it equivalent sequential data structure} is a sequential version +of a concurrent data structure that can be used to express correctness +properties by relating executions of the original concurrent data +structure with executions of the equivalent sequential data structure. +The equivalent sequential data structure is often simpler and can in +many cases simply use existing well-tested implementations in the +STL library. + +An execution {\it history} is a total order of method invocations and +responses. A {\it sequential history} is one where all invocations are +followed by the corresponding responses immediately. A concurrent +execution is correct if its behavior is consistent with its equivalent +sequential history replayed on the equivalent sequential data +structure. A concurrent object is linearizable if for all executions: +\begin{enumerate} +\item Each method call appears to take effect instantaneously at some + point between its invocation and response. + +\item The invocations and responses can be reordered to yield a + sequential history under the rule that an invocation cannot be + reordered before the preceding responses. + +\item The concurrent execution yields the same behavior as + the sequential history. +\end{enumerate} + +A weaker variation of linearization is {\it sequential +consistency}\footnote{It is important to note that the term sequential +consistency in the literature is applied to both the consistency model +that data structures expose clients to as well as the guarantees that +the underlying memory system provides for load and store operations.}. +Sequential consistency only requires that there exist a sequential +history that is consistent with the {\it program order} (the intra-thread +order). This ordering does not need to be consistent with the order +that the operations were actually issued in. + +Line-Up~\cite{lineup}, Paraglider~\cite{VechevMCLinear}, and +VYRD~\cite{vyrd} that leverage linearizability to test concurrent data +structures. {\bf Unfortunately, efficient implementations of many +common data structures, e.g., RCU~\cite{rcu}, MS +Queue~\cite{lockfreequeue}, ..., for the C/C++ memory model are +neither linearizable nor sequentially consistent! Thus previous tools +cannot check such data structures under the C/C++ memory model.} + +\mysubsection{New Challenges from the C/C++ Memory Model} +\label{sec:introNewChallenges} + +The C/C++ memory model brings the following two key challenges that prevent the application of previous approaches to checking concurrent data structures to this setting: + +\squishcount +\item {\bf Relaxed Executions Break Existing Data Structure Consistency Models:} +C/C++ data structures often expose clients to weaker (non-SC) +behaviors to gain performance. A common guarantee is to provide +happens-before synchronization between operations that implement +updates and the operations that read those updates. These data +structures often do not guarantee that different threads observe updates in the +same order --- in other words the data structures may expose clients +to weaker consistency models than sequential consistency. For +example, even when one uses the relatively strong \code{acquire} +and \code{release} memory orderings in C++, it is possible for two +different threads to observe two stores happening in +different orders, i.e., executions can fail the IRIW test. +Thus many data structure legitimately admit executions for which there +are no sequential histories that preserve program order. + +Like many other relaxed memory models, the C/C++ memory model does +not include a total order over all memory operations, thus even +further complicating the application of traditional approaches to correctness, +e.g., linearization cannot be applied. In +particular the approaches that relate the behaviors of concurrent data +structures to analogous sequential data structures break down due to +the absence of a total ordering of the memory operations. While many +of the dynamic tools~\cite{cdschecker,relacy} for exploring the +behavior of code under relaxed models do as a practical matter +print out an execution in some order, this order is to some degree arbitrary as relaxed memory models +generally make it possible for a data structure operation to see the +effects of operations that appear later in any such an order (e.g., a load +can read from a store that appears later in the order). Instead of a total order, the C/C++ +memory model is formulated as a graph of memory operations +with several partial orders defined in this graph. + +\item {\bf Constraining Reorderings (Specifying Synchronization Properties):} +Synchronization\footnote{Synchronization here is not mutual exclusion, + but rather a lower-level property that captures which stores must be + visible to a thread. In other words, it constrains which + reorderings can be performed by a processor or compiler.} in C/C++ + provides an ordering between memory operations to different + locations. Concurrent data structures must establish + synchronization or they potentially expose their users to highly + non-intuitive behavior that is likely to break client code. For + example, consider the case of a concurrent queue that does not + establish synchronization between enqueue and dequeue operations. + If one thread initializes the fields of an object and then enqueues + a reference to that object in such a queue and a second thread + dequeues the reference and uses it to read the fields of that + object, the second thread could fail to see the initializing writes. + This surprising behavior could occur if the compiler or CPU could + reorder the initializing writes to be executed after the enqueue + operation. If the fields are non-atomic, such loads are considered + data races and violate the data race free requirement of the C/C++ + language standard and thus the program has no semantics. + +The C/C++ memory model formalizes synchronization in terms of a {\it + happens-before} relation. The C/C++ happens-before relationship +is a partial order over memory accesses. If memory access {\it x} +happens before memory access {\it y}, it means that the effects of +{\it x} must be ordered before the effects of {\it y}. +\countend + +\begin{figure} + \centering +\vspace{-.3cm} + \includegraphics[scale=0.35]{figures/specworkflow} +\vspace{-.4cm} + \caption{\label{fig:specworkflow} \TOOL system overview} +\vspace{-.4cm} +\end{figure} + +\mysubsection{Specification Language and Tool Support} + +Figure~\ref{fig:specworkflow} presents an overview of the \TOOL system. After +implementing a concurrent data structure, developers annotate their code +with a \TOOL specification. To test their implementation, +developers compile the data structure with the \TOOL specification +compiler to extract the specification and generate a program that is instrumented with specification checks. + Then, developers compile the instrumented program with a +standard C/C++ compiler. Finally, developers run the binary under +the \TOOL checker. \TOOL +then exhaustively explores the behaviors of the specific unit test and +generates diagnostic reports for any executions that violate the specification. + +\mysubsection{Contributions} + +This paper makes the following contributions: + +\begin{itemize} +\item {\bf Specification Language:} It introduces a specification + language that enables developers to write specifications of + concurrent data structures developed for a relaxed memory model in a + simple fashion that captures the key correctness properties. Our + specification language is the first to our knowledge that supports concurrent data + structures that use C/C++ atomic operations. + +\item {\bf A Technique to Relate Concurrent Executions to Sequential Executions:} It presents an approach to order the memory operations for the + C/C++ model, which lacks a definition of a trace and for which one + generally cannot even construct a total order of atomic operations + that is consistent with the program order. The generated + sequential execution by necessity does not always maintain program order. + +\item {\bf Synchronization Properties:} It presents constructs + for specifying the happens before relations that a data structure + should establish and tool support for checking these properties and + exposing synchronization related bugs. + +\item {\bf Tool for Checking C/C++ Data Structures Against Specifications:} \TOOL is the first tool to our knowledge that can check concurrent data structures that exhibit relaxed behaviors against specifications that are specified in terms intuitive sequential executions. + +\item {\bf Evaluation:} It shows that the \TOOL specification language can express key correctness properties for a +set of real-world concurrent data structures, that our tool can detect + bugs, and that our tool can unit test real world data structures + with reasonable performance. +\end{itemize} + diff --git a/correctness-model/writeup/.#introduction.tex.1.26 b/correctness-model/writeup/.#introduction.tex.1.26 new file mode 100644 index 0000000..33c8499 --- /dev/null +++ b/correctness-model/writeup/.#introduction.tex.1.26 @@ -0,0 +1,223 @@ +\mysection{Introduction}\label{sec:introduction} + +Concurrent data structure design can improve scalability by supporting +multiple simultaneous operations, reducing memory coherence traffic, and reducing the time taken by an +individual data structure operation. Researchers have developed many concurrent +data structure designs with these goals~\cite{rcu, +lockfreequeue}. Concurrent data structures often use sophisticated techniques +including low-level atomic instructions (e.g., compare and +swap), careful reasoning about the order of loads and +stores, and fine-grained locking. For example, while the standard Java hash +table implementation can limit scalability to a handful of cores, +more sophisticated concurrent hash tables can scale to many hundreds of +cores~\cite{javaConcurrentHashMap}. + +The C/C++ standard committee extended the C and C++ languages with +support for low-level atomic operations in the C/C++11 +standard~\cite{cpp11spec, c11spec, boehmpldi} to enable developers to +write portable implementations of concurrent data structures. To support the relaxations typically performed by compilers and processors, +the +C/C++ memory model provides weaker semantics than sequential +consistency~\cite{scmemorymodel} and as a result, correctly using +these operations is challenging. Developers must not only +reason about potential interleavings, but also about how the processor +and compiler might reorder memory operations. Even experts make +subtle errors when reasoning about such memory models. + +Researchers have developed tools for exploring the behavior of code +under the C/C++ model including \cdschecker~\cite{cdschecker}, {\sc +Cppmem}~\cite{c11popl}, and Relacy~\cite{relacy}. These tools explore +behaviors that are allowed under the C/C++ memory model. While these +tools can certainly be useful for exploring executions, they can be +challenging to use for testing as they don't provide support (other +than assertions) for specifying the behavior of data structures. +Using assertions can be challenging as different interleavings or +reorderings legitimately produce different behaviors, and it can be +very difficult to code assertions to check the output of a test case under for an +arbitrary (unknown) execution. + +This paper presents \TOOL, a specification language and specification +checking tool that is designed to be used in conjunction with model +checking tools. We have implemented it as a plugin for +the \cdschecker model checker. + +\mysubsection{Background on Linearizability} + +Researchers have developed several techniques for specifying +correctness properties of concurrent data structures written for strong memory +models. While these techniques cannot handle the behaviors typically +exhibited by relaxed data structure implementations, they provide insight into +intuitive approaches to specifying concurrent data structure behavior. + +One approach for specifying the correctness of concurrent data +structures is in terms of equivalent sequential executions of either +the concurrent data structure or a simplified sequential version. The +problem then becomes how do we map a concurrent execution to an equivalent +sequential execution? A common criterion is {\it + linearizability} --- linearizability simply states that a concurrent +operation can be viewed as taking effect at some time between its +invocation and its return (or response)~\cite{linearizableref}. + +An {\it equivalent sequential data structure} is a sequential version +of a concurrent data structure that can be used to express correctness +properties by relating executions of the original concurrent data +structure with executions of the equivalent sequential data structure. +The equivalent sequential data structure is often simpler, and in +many cases one can simply use existing well-tested implementations from the +STL library. + +An execution {\it history} is a total order of method invocations and +responses. A {\it sequential history} is one where all invocations are +followed by the corresponding responses immediately. A concurrent +execution is correct if its behavior is consistent with its equivalent +sequential history replayed on the equivalent sequential data +structure. A concurrent object is linearizable if for all executions: +\begin{enumerate} +\item Each method call appears to take effect instantaneously at some + point between its invocation and response. + +\item The invocations and responses can be reordered to yield a + sequential history under the rule that an invocation cannot be + reordered before the preceding responses. + +\item The concurrent execution yields the same behavior as + the sequential history. +\end{enumerate} + +A weaker variation of linearization is {\it sequential +consistency}\footnote{It is important to note that the term sequential +consistency in the literature is applied to both the consistency model +that data structures expose clients to as well as the guarantees that +the underlying memory system provides for load and store operations.}. +Sequential consistency only requires that there exist a sequential +history that is consistent with the {\it program order} (the intra-thread +order). This ordering does not need to be consistent with the order +that the operations were actually issued in. + +Line-Up~\cite{lineup}, Paraglider~\cite{VechevMCLinear}, and +VYRD~\cite{vyrd} leverage linearizability to test concurrent data +structures. {\bf Unfortunately, efficient implementations of many +common data structures, e.g., RCU~\cite{rcu}, MS +Queue~\cite{lockfreequeue}, ..., for the C/C++ memory model are +neither linearizable nor sequentially consistent! Thus previous tools +cannot check such data structures under the C/C++ memory model.} + +\mysubsection{New Challenges from the C/C++ Memory Model} +\label{sec:introNewChallenges} + +The C/C++ memory model brings the following two key challenges that prevent the application of previous approaches to checking concurrent data structures to this setting: + +\squishcount +\item {\bf Relaxed Executions Break Existing Data Structure Consistency Models:} +C/C++ data structures often expose clients to weaker (non-SC) +behaviors to gain performance. A common guarantee is to provide +happens-before synchronization between operations that implement +updates and the operations that read those updates. These data +structures often do not guarantee that different threads observe updates in the +same order --- in other words the data structures may expose clients +to weaker consistency models than sequential consistency. For +example, even when one uses the relatively strong \code{acquire} +and \code{release} memory orderings in C++, it is possible for two +different threads to observe two stores happening in +different orders, i.e., executions can fail the IRIW test. +Thus many data structure legitimately admit executions for which there +are no sequential histories that preserve program order. + +Like many other relaxed memory models, the C/C++ memory model does +not include a total order over all memory operations, thus even +further complicating the application of traditional approaches to correctness, +e.g., linearization cannot be applied. In +particular the approaches that relate the behaviors of concurrent data +structures to analogous sequential data structures break down due to +the absence of a total ordering of the memory operations. While many +of the dynamic tools~\cite{cdschecker,relacy} for exploring the +behavior of code under relaxed models do as a practical matter +print out an execution in some order, this order is to some degree arbitrary as relaxed memory models +generally make it possible for a data structure operation to see the +effects of operations that appear later in any such an order (e.g., a load +can read from a store that appears later in the order). Instead of a total order, the C/C++ +memory model is formulated as a graph of memory operations +with several partial orders defined in this graph. + +\item {\bf Constraining Reorderings (Specifying Synchronization Properties):} +Synchronization\footnote{Synchronization here is not mutual exclusion, + but rather a lower-level property that captures which stores must be + visible to a thread. In other words, it constrains which + reorderings can be performed by a processor or compiler.} in C/C++ + provides an ordering between memory operations to different + locations. Concurrent data structures must establish + synchronization or they potentially expose their users to highly + non-intuitive behavior that is likely to break client code. For + example, consider the case of a concurrent queue that does not + establish synchronization between enqueue and dequeue operations. Consider the + following sequence of operations: (1) thread A initializes the fields of a new + object X; (2) thread A enqueues the reference to X in such a queue + (3) thread B dequeues the reference to X; (4) thread B reads the + fields of X through the dequeued reference. In (4), thread B could fail to see the + initializing writes from (1). + This surprising behavior could occur if the compiler or CPU could + reorder the initializing writes to be executed after the enqueue + operation. If the fields are non-atomic, such loads are considered + data races and violate the data race free requirement of the C/C++ + language standard and thus the program has no semantics. + +The C/C++ memory model formalizes synchronization in terms of a {\it + happens-before} relation. The C/C++ happens-before relationship +is a partial order over memory accesses. If memory access {\it x} +happens before memory access {\it y}, it means that the effects of +{\it x} must be ordered before the effects of {\it y}. +\countend + +\begin{figure} + \centering +\vspace{-.3cm} + \includegraphics[scale=0.35]{figures/specworkflow} +\vspace{-.4cm} + \caption{\label{fig:specworkflow} \TOOL system overview} +\vspace{-.4cm} +\end{figure} + +\mysubsection{Specification Language and Tool Support} + +Figure~\ref{fig:specworkflow} presents an overview of the \TOOL system. After +implementing a concurrent data structure, developers annotate their code +with a \TOOL specification. To test their implementation, +developers compile the data structure with the \TOOL specification +compiler to extract the specification and generate a program that is instrumented with specification checks. + Then, developers compile the instrumented program with a +standard C/C++ compiler. Finally, developers run the binary under +the \TOOL checker. \TOOL +then exhaustively explores the behaviors of the specific unit test and +generates diagnostic reports for any executions that violate the specification. + +\mysubsection{Contributions} + +This paper makes the following contributions: + +\begin{itemize} +\item {\bf Specification Language:} It introduces a specification + language that enables developers to write specifications of + concurrent data structures developed for a relaxed memory model in a + simple fashion that captures the key correctness properties. Our + specification language is the first to our knowledge that supports concurrent data + structures that use C/C++ atomic operations. + +\item {\bf A Technique to Relate Concurrent Executions to Sequential Executions:} It presents an approach to order the memory operations for the + C/C++ model, which lacks a definition of a trace and for which one + generally cannot even construct a total order of atomic operations + that is consistent with the program order. The generated + sequential execution by necessity does not always maintain program order. + +\item {\bf Synchronization Properties:} It presents constructs + for specifying the happens before relations that a data structure + should establish and tool support for checking these properties and + exposing synchronization related bugs. + +\item {\bf Tool for Checking C/C++ Data Structures Against Specifications:} \TOOL is the first tool to our knowledge that can check concurrent data structures that exhibit relaxed behaviors against specifications that are specified in terms intuitive sequential executions. + +\item {\bf Evaluation:} It shows that the \TOOL specification language can express key correctness properties for a +set of real-world concurrent data structures, that our tool can detect + bugs, and that our tool can unit test real world data structures + with reasonable performance. +\end{itemize} + diff --git a/correctness-model/writeup/.#memorymodel.tex.1.1 b/correctness-model/writeup/.#memorymodel.tex.1.1 new file mode 100644 index 0000000..f60d9da --- /dev/null +++ b/correctness-model/writeup/.#memorymodel.tex.1.1 @@ -0,0 +1,101 @@ +\section{C/C++ Memory Model}\label{sec:memorymodel} + +We next briefly summarize key aspects of the C/C++ memory model. The memory model describes a series of atomic operations and the +corresponding allowed behaviors of programs that utilize them. + +Any operation on an atomic object will have one of six \textit{memory orders}, +each of which falls into one or more of the following categories. + +\begin{description} + + \item[relaxed:] + \code{memory\_order\_relaxed} -- + weakest memory ordering + + \item[release:] + \code{memory\_order\_release}, + \code{memory\_order\_acq\_rel}, and + \code{memory\_order\_seq\_cst} -- + a store-release may form release/consume or release/acquire synchronization + + \item[consume:\footnotemark{}] + \code{memory\_order\_consume} -- + a load-consume may form release/consume synchronization + + \footnotetext{Consume is not broadly supported by compilers at + this point due to specifying data dependencies. We treat consumes as acquires.} + + \item[acquire:] + \code{memory\_order\_acquire}, + \code{memory\_order\_acq\_rel}, and + \code{memory\_order\_seq\_cst} -- + a load-acquire may form release/acquire synchronization + + \item[seq-cst:] + \code{memory\_order\_seq\_cst} -- + strongest memory ordering + +\end{description} + +\subsection{Relations}\label{sec:model-relations} + +The C/C++ memory model expresses program behavior in the form +of binary relations or orderings. The following subsections will briefly +summarize the relevant relations. Much of this discussion resembles the preferred model from the +formalization in \cite{c11popl}. + +\mypara{\bf Sequenced-Before:} +The order of program operations within a single thread of execution establishes an intra-thread +\textit{sequenced-before} (\reltext{sb}) relation. +Note that while some operations in C/C++ provide no +intra-thread ordering---the equality operator (\code{==}), for example---we +ignore this detail and assume that \reltext{sb} totally orders all operations in +a thread. + +\mypara{\bf Reads-From:} +The \textit{reads-from} (\reltext{rf}) relation consists of store-load pairs $(X, Y)$ +such that $Y$ reads its value from the effect of $X$---or $X \relation{rf} Y$. In the +C/C++ memory model, this relation is non-trivial, as a given load operation may +read from one of many potential stores in the program execution. + +\mypara{\bf Synchronizes-With:} +The \textit{synchronizes-with} (\reltext{sw}) relation captures +synchronization that occurs when certain atomic operations interact across two +threads. For instance, release/acquire synchronization occurs between a pair of +atomic operations on the same object: a store-release $X$ and a load-acquire +$Y$. If $Y$ reads from $X$, then $X$ synchronizes with $Y$---or $X +\relation{sw} Y$. Synchronization also occurs between consecutive unlock and +lock operations on the same mutex, between thread creation and the first event +in the new thread, and between the last action of a thread and the completion of +a thread-join operation targeting that thread. + +\mypara{\bf Happens-Before:} +In \TOOL, we avoid consume operations, and so the \textit{happens-before} +(\reltext{hb}) relation is simply the transitive closure of \reltext{sb} and +\reltext{sw}. + +The \reltext{hb} relation restricts the stores that loads can read +from. For example, if we have two stores $X$ and $Y$ and +a load $Z$ to the same memory location and $X \relation{hb} Y +\relation{hb} Z$, then $Z$ cannot read from $X$. + +\mypara{\bf Sequential Consistency:} +All seq-cst operations in a program execution form a +total ordering (\reltext{sc}) so that, for instance, a seq-cst load may not read +from a seq-cst store prior to the most recent store (to the same location) in +the \reltext{sc} ordering, nor from any store that happens before that store. +The \reltext{sc} order must be consistent with \reltext{hb}. + +\mypara{\bf Modification Order:} +Each atomic object in a program execution has an associated \textit{modification order} +(\reltext{mo})---a total order of all stores to that object---which +informally represents a memory-coherent ordering in which those stores may be observed by +the rest of the program. +Note that +in general the modification orders for all objects cannot be combined +to form a consistent total ordering. We use $X \relation{mo} Y$ to represent the +fact that $X$ is modification ordered before $Y$. + + + + diff --git a/correctness-model/writeup/.#specification.tex.1.47 b/correctness-model/writeup/.#specification.tex.1.47 new file mode 100644 index 0000000..cb2aa21 --- /dev/null +++ b/correctness-model/writeup/.#specification.tex.1.47 @@ -0,0 +1,419 @@ +\mysection{Specification Language Design}\label{sec:specification} + +We begin by briefly overviewing \TOOL's basic approach. The \TOOL +specification language specifies the correctness properties for +concurrent data structures by establishing a correspondence with an +equivalent sequential data structure, which requires: (1) defining the +equivalent sequential data structure; (2) establishing an equivalent +sequential history; (3) relating the behavior of the concurrent +execution to the sequential history; and (4) specifying the +synchronization properties between method invocations. The \TOOL +specification language has the following key components: + +\mypara{\bf 1. Equivalent Sequential Data Structure:} This component defines a +sequential data structure against which the concurrent data structure +will be checked. + +\mypara{\bf 2. Defining the Equivalent Sequential History:} +For real-world data structures, generally there exist two types of API methods, +\textit{primitive} API methods and \textit{aggregate} API methods. Take +concurrent hashtables as an example, it provides primitive API methods +such as \code{put} and \code{get} that implement the core functionality of the +data structure, and it also has aggregate API methods such +as \code{putAll} that calls primitive API methods internally. From our +experience working with our benchmark set, it is generally possible to +generate a sequential history of primitive API method invocations for +real-world data structures as they generally serialize on a single memory +location --- while it is possible to observe partially completed aggregate +API method invocations. Therefore, our specifications will focus on the +correctness of primitive API methods. + +Borrowing from VYRD~\cite{vyrd} the concept of commit points, we allow +developers to specify \textit{ordering points} --- the specific atomic +operations between method invocation and response events that are used for +ordering method calls. Ordering points are a generalization of the commit +points as they also use memory operations that do not commit the data structure +operation to order method calls. Developers then specify ordering points to +generate the equivalent sequential history. + +\mypara{\bf 3. Specifying Correct Behaviors:} Developers use +the \TOOL specification to provide a set of \textit{side effects} +and \textit{assertions} to describe the desired behavior of each API +method. Side effects capture how a method updates the equivalent +sequential data structure. Assertions include both preconditions and +postconditions which specify what conditions each method must satisfy +before and after the method call, respectively. The specification of +the side effects and assertions may make use of the states of the +equivalent sequential data structure, meaning that these components +can access the internal variables and methods of the equivalent sequential data +structure. Additionally, they can reference the values available at +the method invocation and response, i.e., the parameter values and the +return value. + +\mypara{\bf 4. Synchronization:} The synchronization specification +describes the desired happens before relations between API method +invocations. \TOOL specifications allow developers to specify the +properties at the abstraction of methods. This makes specifications +cleaner, more understandable, and less error-prone because the +properties do not rely on low-level implementation +details. Moreover, \TOOL specifications allow developers to attach +conditions to methods when specifying synchronization properties so +that a method call might synchronize with another only under a +specific condition. For example, consider a spin lock with +a \code{try\_lock()} method, we need to specify that only a +successful \code{try\_lock()} method invocation must synchronize with the +previous \code{unlock()} method invocation. + +Figure~\ref{fig:speccfg} presents the grammar for the \TOOL specification +language. The grammar defines three types of specification annotations: +\textit{structure annotations}, \textit{method annotations}, and +\textit{ordering point annotations}. +In the grammar, \textit{code} means legal C/C++ source code; and +\textit{label} means legal C/C++ variable name. +Annotations are embedded in C/C++ comments. +This format does not affect the semantics of the original +program and allows for the same source to be used by both a standard C/C++ +compiler to generate production code and for the \TOOL specification compiler to +extract the \TOOL specification from the comments. We discuss these constructs in more detail throughout the remainder of this section. + +\begin{figure}[!tb] +\vspace{-.2cm} +{\scriptsize + \begin{eqnarray*} + \textit{\textbf{structureSpec}} & \rightarrow & \code{"@Structure\_Define"}\\ + && \textit{structureDefine} \ \ \ (\textit{happensBefore})?\\ + \textit{structureDefine} & \rightarrow & +(\code{"@DeclareStruct:"} \ + \textit{code})\text{*}\\ + && \code{"@DeclareVar:"} \ \textit{code}\\ + && \code{"@InitVar:"} \ \textit{code}\\ + && (\code{"@DefineFunc:"} \ \textit{code})\text{*}\\ + \textit{happensBefore} & \rightarrow & \code{"@Happens\_Before:"} \ + (\textit{invocation} \ \code{"->"} \ + \textit{invocation})\textsuperscript{+}\\ + \textit{invocation} & \rightarrow & \textit{label} \ (\ \ \code{"("} \ + \textit{label} \code{")"} \ \ )?\\ + \textit{\textbf{methodSpec}} & \rightarrow & \code{"@Method:"} \ \textit{label}\\ + && \code{"@Ordering\_Point\_Set:"} \ \textit{label} \ (\code{"|"} \ + \textit{label})\text{*}\\ + && (\code{"@HB\_Condition:"} \ \textit{label} \ \code{"::"} \ \textit{code})\text{*}\\ + && (\code{"@ID:"} \ \textit{code})?\\ + && (\code{"@PreCondition:"} \ \textit{code})?\\ + && (\code{"@SideEffect:"} \ \textit{code})?\\ + && (\code{"@PostCondition:"} \ \textit{code})?\\ + \textit{\textbf{potentialOP}} & \rightarrow & \code{"@Potential\_Ordering\_Point:"} \ \textit{code}\\ + && \code{"@Label:"} \ \ \textit{label}\\ + \textit{\textbf{opCheck}} & \rightarrow & \code{"@Ordering\_Point\_Check:"} \ \textit{code}\\ + && \code{"@Potential\_Ordering\_Point\_Label:"} \ \textit{label}\\ + && \code{"@Label:"} \ \ \textit{label}\\ + \textit{\textbf{opLabelCheck}} & \rightarrow & \code{"@Ordering\_Point\_Label\_Check:"} \ \textit{code}\\ + && \code{"@Label:"} \ \ \textit{label}\\ + \textit{\textbf{opClear}} & \rightarrow & \code{"@Ordering\_Point\_Clear:"} \ \textit{code}\\ + && \code{"@Label:"} \ \ \textit{label}\\ + \textit{code} & \rightarrow & \code{} + \end{eqnarray*}} +\vspace{-.7cm} +\caption{\label{fig:speccfg}Grammar for \TOOL specification language} +\vspace{-.3cm} +\end{figure} + + +\mysubsection{Example} + +To make the \TOOL specification language more concrete, +Figure~\ref{fig:rcuSpecExample} presents the annotated RCU example. We +use {\it ordering points} to order the method invocations to construct +the equivalent sequential history for the concurrent execution. We +first specify the ordering points for the \code{read} +and \code{update} methods to help define the equivalent sequential +history. For the +\code{read} method, the load access of the \code{node} field in +Line~\ref{line:rcuSpecReadLoad} is the only operation that can be an ordering +point. For the \code{update} method, there exists more than one atomic +operation, however they only serve as an ordering point when it successfully uses the CAS operation +to update the \code{node} field. Thus, we specify in +Line~\ref{line:rcuSpecUpdateOPBegin} that the CAS operation should be the +ordering point for the update operation when \code{succ} is \code{true}. We +associate the methods with the two ordering points in +Line~\ref{line:rcuSpecReadInterfaceOPSet} and +\ref{line:rcuSpecUpdateInterfaceOPSet}. + +Next, we specify the equivalent sequential data structure for this RCU implementation. +Line~\ref{line:declVar} declares two integer fields \code{\_data} and +\code{\_version} as the internal states of the equivalent sequential RCU, and +Line~\ref{line:initVar} initializes both variables to \code{0}. + +We then specify the correct behaviors of the equivalent sequential +history by defining the side effects and assertions for +the \code{read} and \code{update} methods. Side effects specify how +to perform the corresponding operation on the equivalent sequential +data structure. For example, +Line~\ref{line:rcuSpecUpdateInterfaceSideEffect} specifies the side +effects of the \code{update} to the sequential states. +Assertions specify properties that should be true of the concurrent data structure execution. +For example, Line~\ref{line:rcuSpecReadInterfacePostCondition} +specifies that the \code{read} method should satisfy the postcondition that the +returned \code{data} and \code{version} fields should have consistent values +as the internal state of the equivalent sequential data structure. + +Our implementation of the RCU data structure is designed to establish +synchronization between the \code{update} method invocation and the later +\code{read} or \code{update} method invocation. This is important, for +example, if a client thread were to update an array index and use the RCU data +structure to communicate the updated index to a second client thread. Without +synchronization, the second client thread may not see the update to the array +index. Besides, the synchronization between two \code{update} calls ensures that +later \code{read} will see the most updated values. +In order to specify the synchronization, we associate \code{read} and \code{update} methods with method call identifiers +(Line~\ref{line:rcuSpecReadInterfaceID} and Line~\ref{line:rcuSpecUpdateInterfaceID}), which for this example is the value of the +\code{this} pointer. +Together these annotations ensure that every +API method call on the same RCU object will have the same method call ID. +Line~\ref{line:rcuSpecHB} then specifies +the synchronization property for the RCU --- any \code{update} method invocation +should synchronize with all later \code{read} and \code{update} method invocations that have the +same method call identifier as that \code{update} method. This guarantees that +\code{update} calls should synchronize with the \code{read} and \code{update} +calls of the same RCU object. + + +\begin{figure}[h!] +\vspace{-.2cm} +\begin{lstlisting}[xleftmargin=6.0ex] +class RCU { + /** @Structure_Define: + @DeclareVar:/*@ \label{line:declVar} @*/ int _data, _version; + @InitVar:/*@ \label{line:initVar} @*/ _data = 0; _version = 0; + @Happens_Before: Update->Read Update->Update*//*@ \label{line:rcuSpecHB} @*/ + atomic node; + public: + RCU() { + Node *n = new Node; + n->data = 0; + n->version = 0; + atomic_init(&node, n); + } + /** @Interface: Read/*@ \label{line:rcuSpecReadInterfaceLabel} @*/ + @Ordering_Point_Set: Read_Point/*@ \label{line:rcuSpecReadInterfaceOPSet} @*/ + @ID: this/*@ \label{line:rcuSpecReadInterfaceID} @*/ + @PostCondition:/*@ \label{line:rcuSpecReadInterfacePostCondition} @*/ + _data == *data && _version == *version *//*@ \label{line:rcuSpecReadInterfaceEnd} @*/ + void read(int *data, int *version) { + Node *res = node.load(mo_acquire);/*@ \label{line:rcuSpecReadLoad} @*/ + /** @Ordering_Point_Label_Check: true/*@ \label{line:rcuSpecReadOPBegin} @*/ + @Label: Read_Point */ + *data = res->data; + *version = res->version; + } + /** @Interface: Update + @Ordering_Point_Set: Update_Point/*@ \label{line:rcuSpecUpdateInterfaceOPSet} @*/ + @ID: this/*@ \label{line:rcuSpecUpdateInterfaceID} @*/ + @SideEffect: _data = data; _version++; *//*@ \label{line:rcuSpecUpdateInterfaceSideEffect} @*/ + void update(int data) { + bool succ = false; + Node *newNode = new Node;/*@ \label{line:rcuSpecUpdateAlloc} @*/ + Node *prev = node.load(mo_acquire);/*@ \label{line:rcuSpecUpdateLoad} @*/ + do { + newNode->data = data; + newNode->version = prev->version + 1; + succ = node.compare_exchange_strong(prev,/*@\label{line:rcuSpecUpdateCAS} @*/ + newNode, mo_acq_rel, mo_acquire); + /** @Ordering_Point_Label_Check: succ/*@ \label{line:rcuSpecUpdateOPBegin} @*/ + @Label: Update_Point */ + } while (!succ); + } +}; + +\end{lstlisting} +\vspace{-.2cm} +\caption{\label{fig:rcuSpecExample}Annotated RCU specification example} +\vspace{-.3cm} +\end{figure} + +\subsection{Defining the Equivalent Sequential History} + +While defining the equivalent sequential history for concurrent executions is well studied in the context of the +SC memory model, optimizations that developers typically use in the +context of weaker memory models create the following new challenges when +we generate the equivalent sequential history using ordering points. + +\begin{itemize} +\item {\bf Absence of a Meaningful Trace or Total Order:} +For the SC memory model, an execution can be represented by a simple +interleaving of all the memory operations, where each load operation reads from +the last store operation to the same location in the trace. However, under a +relaxed memory model like C/C++11, the interleaving does not uniquely define an +execution because a given load operation can potentially read from many +different store operations in the interleaving. Therefore, we have to rely on +the intrinsic relations between ordering points such as \textit{reads from} and +\textit{modification order} to order method calls. + +\item {\bf Lack of Program Order Preserving Sequential History:} +Moreover, as discussed in Section~\ref{sec:exampleMotivation}, in +general it is not possible to arrange an execution in any totally +ordered sequential history that preserves program order. + +A key insight is that many concurrent data structures' API methods +have a commit point, which is a single memory operation that makes the +update visible to other threads and that also serves as an ordering +point. When two data structure operations have a dependence, it is +often the case that their respective commit points are both +conflicting accesses to the same memory location. In this case, the +modification order provided by C/C++ is sufficient to order these +operations since modification order is guaranteed to be consistent +with the happens before relation (and therefore also the sequenced +before relation). + +For cases where the method calls are independent, such as +a \code{put(X, 1)} followed by a +\code{get(Y)} in a hashtable where \code{X} and \code{Y} are different keys, +the lack of an ordering is not a problem since those methods commute. +\end{itemize} + +\mypara{\bf Ordering Point Annotations:} In many cases, it is not +possible to determine whether a given atomic operation is an ordering +point until later in the execution. For example, +some internal methods may be called by multiple API methods. In this +case, the same atomic operation can be identified as a potential +ordering point for multiple API methods, and each API method later has +a checking annotation to verify whether it was a real ordering +point. Therefore, the \TOOL specification separates the definition of ordering +points as follows: +\begin{enumerate} +\vspace{-.1cm} +\item {\code{Potential\_Ordering\_Point} annotation:} The labeling of ordering +points that identifies the location of a potential ordering point. +\vspace{-.2cm} +\item {\code{Ordering\_Point\_Check} annotation:} The checking of ordering +points that checks at a later point whether a potential ordering point was +really an ordering point. +\end{enumerate} +\vspace{-.1cm} + +These two constructs together identify ordering points. For +example, assume that $A$ is an atomic operation that is marked as a +potential ordering point with the label \code{LabelA} under the +condition \code{ConditionA}. The developer would write a +\code{Potential\_Ordering\_Point} annotation with the label +\code{LabelA} and then use the label \code{LabelA} in an +\code{Ordering\_Point\_Check} annotation at a later point. + +The \code{Ordering\_Point\_Label\_Check} annotation combines +the \code{Potential\_Ordering\_Point} and the +\code{Ordering\_Point\_Check} annotations, and makes +specifications simpler for the use case that the ordering point is known immediately. +For example, in Line~\ref{line:rcuSpecReadOPBegin} of +Figure~\ref{fig:rcuSpecExample}, we use the \code{Ordering\_Point\_Label\_Check} annotation to +identify the ordering point for \code{read} because we know the load operation +in Line~\ref{line:rcuSpecReadLoad} is an ordering point at the time it is +executed. + +Some data structure operations may require multiple ordering points. +For example, consider a transaction implementation that first attempts +to lock all of the involved objects (dropping the locks and retrying +if it fails to acquire a lock), performs the updates, and then releases +the locks. To order such transactions in a relaxed memory model, we must consider all of the +locks it acquires and not just the last lock. Thus, we allow a +method invocation to have more than one ordering point, and the +additional ordering points serve to order the operation with respect to +multiple different memory locations. For the transaction example, it +may be necessary to retry the acquisition of locks. To support this +scenario, the \code{Ordering\_Point\_Clear} annotation removes all +previous ordering points when it satisfies a specific condition. + +Moreover, when an API method calls another API method, they can share ordering points. In +that case, \TOOL requires that at that ordering point, the concurrent data +structure should satisfy the precondition and postcondition of both the inner +and outer API methods. + +\subsection{Checking the IO Behavior of Methods} + +With the specified ordering points, \TOOL is able to generate the equivalent +sequential history. Developers then need to define the equivalent sequential +data structure. For example, in Line~\ref{line:declVar} and +~\ref{line:initVar} of the annotated RCU example, we use the +\code{Structure\_define} annotation to define the equivalent sequential RCU +by specifying the internal states as two integers, \code{\_verion} and \code{\_data}. In the +\code{Structure\_define} annotation, developers can also specify definitions for customized +structs and methods for convenience. We design these annotations in such a way +that developers can +write specifications in C/C++ code such that they do not have to learn a new +specification language. + +After defining the internal states and methods of the +equivalent data structure, developers use the \code{SideEffect} annotation to +define the corresponding sequential API methods, which should contain the action +to be performed on the equivalent sequential data structure. For example, in +Line~\ref{line:rcuSpecUpdateInterfaceSideEffect} of the annotated RCU example, +we use \code{SideEffect} to specify that when we execute the \code{update} +method on the equivalent sequential RCU, we should update the +internal states of \code{\_version} and \code{\_data} accordingly. When +the \code{SideEffect} annotation is omitted for a specific API method, it means +that no side effects will happen on the sequential data structure when that +method is called. Take the annotated RCU as an example, the \code{read} has no +side effects on the equivalent sequential RCU. + +With the well-defined equivalent sequential data structure, developers then relate +the generated equivalent sequential history to the equivalent sequential data +structure. In \TOOL, we allow developers to accomplish this by using the +\code{PreCondition} and \code{PostCondition} annotations to specify the +conditions to be checked before and after the API method appears to happen. +For example, Line~\ref{line:rcuSpecReadInterfaceEnd} in the +annotated RCU example means that when \code{read} appears to happen, it should +return the same value as the current interal variables of the equivalent +sequential RCU. +Note that these two annotations contain legitimate C/C++ +expressions that only access the method call parameters, return value and the +internal states of the equivalent sequential data structure. + +\subsection{Checking Synchronization} + +Under a relaxed memory model, compilers and processors can reorder +memory operations and thus the execution can exhibit counter-intuitive +behaviors. The C/C++11 memory model provides the developer with memory ordering that establish synchronization, e.g., \code{acquire}, \code{release}, \code{seq\_cst}. Synchronization +serves to control which reorderings are allowed --- however, +restricting reorderings comes at a runtime cost so developers must +balance complexity against runtime overhead. Checking that data +structures establish proper synchronization is important to ensure +that the data structures can be effectively composed with client code. + +We generalize the notion of happens before to methods as follows. +Method call $c_1$ happens-before method call $c_2$ if the invocation +event of $c_1$ happens before the response event of $c_2$. Note that +by this definition two method calls can both happen before each other +--- an example of this is the \code{barrier} synchronization +construct. For example, for a correctly synchronized queue, we want +an enqueue to happen before the corresponding dequeue, which avoids +the synchronization problems discussed earlier in +Section~\ref{sec:introNewChallenges}. + +In order to flexibly express the synchronization between methods, we associate +API methods with method call identifiers (or IDs) and happens-before conditional guard +expressions. The method +call ID is a C/C++ expression that computes a unique ID for the call, and if it +is omitted, a default value is used. For example, in our RCU example in +Figure~\ref{fig:rcuSpecExample}, method \code{update} and \code{read} both +have the \code{this} pointer of the corresponding RCU object. +The \code{HB\_Condition} component +associates one happens-before conditional guard expression with a unique label. +For one method, multiple conditional guard expressions are allowed to be +defined, and the conditional guard expression can only access the method +instance's argument values and return value. + +After specifying the +method call IDs and the \code{HB\_Condition} labels, developers +can specify the synchronization as ``\code{method1(HB\_condition1)} \code{->} +\code{method2(HB\_condition2)}''. When the \code{HB\_condition} is omitted, it +defaults to \code{true}. The semantics of this expression is that all +instances of calls to \code{method1} that satisfy the conditional guard +expression \code{HB\_condition1} should happen-before all later instances (as determined by ordering points) of +calls to \code{method2} that satisfy the conditional guard expression +\code{HB\_condition2} such that both instances shared the same ID. +The ID and happens-before conditional guard expression are important because they +allow developers to impose synchronization only between specific method +invocations under specific conditions. For example, in +Figure~\ref{fig:rcuSpecExample}, Line~\ref{line:rcuSpecHB} specifies two +synchronization rules, which together mean that the +\code{update} should only establish synchronization with later \code{read} and +\code{update} from +the same RCU object under all circumstances. diff --git a/correctness-model/writeup/CVS/Entries b/correctness-model/writeup/CVS/Entries new file mode 100644 index 0000000..e738c4e --- /dev/null +++ b/correctness-model/writeup/CVS/Entries @@ -0,0 +1,19 @@ +/confstrs-abbrv.bib/1.1/Wed Jan 14 22:53:32 2015// +/confstrs-long.bib/1.1/Wed Jan 14 22:53:32 2015// +/sig-alternate.cls/1.1/Wed Jan 14 22:53:32 2015// +D/figures//// +/makefile/1.2/Fri Jan 23 17:16:19 2015// +/spell.lst/1.1/Fri Jan 23 05:47:30 2015// +/abstract.tex/1.2/Fri Jan 23 22:58:26 2015// +/conclusion.tex/1.2/Fri Jan 23 22:58:26 2015// +/related.tex/1.2/Fri Jan 23 22:58:26 2015// +/speccfg.tex/1.1/Fri Jan 23 22:58:26 2015// +/technical.tex/1.1/Fri Jan 23 22:58:26 2015// +/paper.tex/1.6/Fri Jan 23 23:30:25 2015// +/paper.bib/1.3/Sat Jan 24 00:47:32 2015// +/memorymodel.tex/1.8/Sat Jan 24 01:51:58 2015// +/introduction.tex/1.36/Sat Jan 24 03:09:04 2015// +/example.tex/1.19/Sat Jan 24 03:36:11 2015// +/specification.tex/1.59/Sat Jan 24 04:47:54 2015// +/implementation.tex/1.16/Sat Jan 24 05:04:07 2015// +/evaluation.tex/1.21/Sat Jan 24 05:35:18 2015// diff --git a/correctness-model/writeup/CVS/Repository b/correctness-model/writeup/CVS/Repository new file mode 100644 index 0000000..e5a4afa --- /dev/null +++ b/correctness-model/writeup/CVS/Repository @@ -0,0 +1 @@ +Papers/15/issta.cdsspec diff --git a/correctness-model/writeup/CVS/Root b/correctness-model/writeup/CVS/Root new file mode 100644 index 0000000..cd8b2f1 --- /dev/null +++ b/correctness-model/writeup/CVS/Root @@ -0,0 +1 @@ +:ext:peizhaoo@plrg.eecs.uci.edu:/home/cvs/Papers diff --git a/correctness-model/writeup/abstract.tex b/correctness-model/writeup/abstract.tex new file mode 100644 index 0000000..7bf3748 --- /dev/null +++ b/correctness-model/writeup/abstract.tex @@ -0,0 +1,19 @@ +Concurrent data structures often provide better performance on +multi-core platforms, but are significantly more difficult to design and +verify than their sequential counterparts. The C/C++11 standard +introduced a weak language memory model supporting low-level atomic +operations such as compare and swap (CAS). While these +atomic operations can significantly improve the performance +of concurrent data structures, programming at this level +introduces +non-intuitive behaviors that significantly increase the difficulty of +developing code. + +In this paper, we present \TOOL, a specification language checker that +allows developers to write simple specifications for low-level +concurrent data structures that make use of C/C++11 atomics and check +the correctness of concurrent data structures against these +specifications. \TOOL is designed to be used in conjunction with +model checking tools and we have implemented it as a plugin +to \cdschecker. We have evaluated \TOOL by annotating and checking +several concurrent data structures. diff --git a/correctness-model/writeup/conclusion.tex b/correctness-model/writeup/conclusion.tex new file mode 100644 index 0000000..ae07691 --- /dev/null +++ b/correctness-model/writeup/conclusion.tex @@ -0,0 +1,11 @@ +\mysection{Conclusion\label{sec:conclusion}} + +The \TOOL specification language and checking system makes it easier +to unit test concurrent data structures written for the C/C++11 memory +model. It extends and modifies classic approaches to defining the +desired behaviors of concurrent data structures with respect to +sequential versions of the same data structure to +apply to the C/C++ memory model. Our evaluation shows +that the approach can be used to specify and test correctness +properties for a range of data structures including a +lock-free hashtable, work-stealing deque, queues and locks. diff --git a/correctness-model/writeup/confstrs-abbrv.bib b/correctness-model/writeup/confstrs-abbrv.bib new file mode 100644 index 0000000..ef42052 --- /dev/null +++ b/correctness-model/writeup/confstrs-abbrv.bib @@ -0,0 +1,173 @@ +@string{mspc12="MSPC"} +@string{ppdp11="PPDP"} +@string{spin04="SPIN"} +@string{spin07="SPIN"} +@string{spin08="SPIN"} +@string{spin09="SPIN"} +@string{ijpp="IJPP"} +@string{jlap="JLAP"} +@string{fcad10="FCAD"} +@strign{sc="SC"} +@string{iwops99="POS"} +@string{podc96="PODC"} +@string{popl73="POPL"} +@string{popl93="POPL"} +@string{popl96="POPL"} +@string{popl97="POPL"} +@string{popl99="POPL"} +@string{popl05="POPL"} +@string{popl07="POPL"} +@string{popl08="POPL"} +@string{popl09="POPL"} +@string{popl11="POPL"} +@string{popl13="POPL"} + +@string{dac07="DAC"} +@string{sosp05="SOSP"} +@string{sosp03="SOSP"} +@string{oopsla02="OOPSLA"} +@string{oopsla03="OOPSLA"} +@string{oopsla04="OOPSLA"} +@string{oopsla05="OOPSLA"} +@string{oopsla06="OOPSLA"} +@string{oopsla07="OOPSLA"} +@string{oopsla08="OOPSLA"} +@string{oopsla09="OOPSLA"} +@string{oopsla10="OOPSLA"} +@string{oopsla13="OOPSLA"} + +@string{hotpar09="HotPar"} +@string{hotpar10="HotPar"} + +@string{atc="ATC"} + +@string{ismm08="ISMM"} +@string{ismm09="ISMM"} +@string{ismm10="ISMM"} + +@string{esop09="ESOP"} + +@string{scool05="SCOOL"} + +@string{paste08="PASTE"} + +@string{ecoop10="ECOOP"} + +@string{osdi04="OSDI"} +@string{osdi08="OSDI"} + +@string{toplas02="TOPLAS"} +@string{toplas09="TOPLAS"} + +@string{pact01="PACT"} +@string{pact08="PACT"} + +@string{cc08="CC"} +@string{cc10="CC"} +@string{cc11="CC"} + +@string{asplos91="ASPLOS"} +@string{asplos02="ASPLOS"} +@string{asplos09="ASPLOS"} + +@string{sc01="SC"} +@string{sc06="SC"} + +@string{spaa98="SPAA"} +@string{toplas="TOPLAS"} +@string{tocs="TOCS"} +@string{tpds="TPDS"} +@string{vee05="VEE"} +@string{vee08="VEE"} +@string{jvm01="JVM"} + +@string{ics95="ICS"} +@string{ics99="ICS"} + +@string{icse05="ICSE"} +@string{icse10="ICSE"} +@string{icse00="ICSE"} + +@string{sc97="SC"} +@string{sc01="SC"} +@string{lcpc08="LCPC"} +@string{lcpc01="LCPC"} +@string{pldi="PLDI"} +@string{pldi93="PLDI"} +@string{pldi94="PLDI"} +@string{pldi95="PLDI"} +@string{pldi96="PLDI"} +@string{pldi97="PLDI"} +@string{pldi98="PLDI"} +@string{pldi99="PLDI"} +@string{pldi00="PLDI"} +@string{pldi01="PLDI"} +@string{pldi02="PLDI"} +@string{pldi03="PLDI"} +@string{pldi04="PLDI"} +@string{pldi05="PLDI"} +@string{pldi06="PLDI"} +@string{pldi07="PLDI"} +@string{pldi08="PLDI"} +@string{pldi09="PLDI"} +@string{pldi10="PLDI"} +@string{pldi11="PLDI"} +@string{pldi12="PLDI"} +@string{pldi13="PLDI"} + +@string{issta11="ISSTA"} +@string{isca90="ISCA"} +@string{isca09="ISCA"} +@string{isca10="ISCA"} +@string{ppopp97="PPoPP"} +@string{ppopp01="PPoPP"} +@string{ppopp07="PPoPP"} +@string{ppopp09="PPoPP"} +@string{ppopp10="PPoPP"} +@string{ppopp11="PPoPP"} +@string{ppopp12="PPoPP"} +@string{ppopp13="PPoPP"} + +@string{cgo10="CGO"} +@string{cgo11="CGO"} + +@string{ispass09="ISPASS"} + +@string{iiswc08="IISWC"} +@string{iiswc09="IISWC"} + +@string{iccl88="ICCL"} + +@string{micro05="MICRO"} +@string{micro10="MICRO"} + +@string{hpdc07="HPDC"} + +@string{cav08="CAV"} +@string{cav07="CAV"} +@string{cav06="CAV"} +@string{cav04="CAV"} + +@string{procomet90="PROCOMET"} +@string{hpca07="HPCA"} +@string{dexa04="DEXA"} + +@string{lics="LICS"} +@string{lncs="LNCS"} +@string{lcpc="LCPC"} +@string{vmcai="VMCAI"} +@string{vmcai09="VMCAI"} +@string{tacas11="TACAS"} + +@string{ieeecmpscieng="IEEE Comput. Sci. Eng."} +@string{higherordersymbcomp="HOSC"} +@string{ibmjrd="IBM J. of Res. Dev."} +@string{ieeetc="TC"} +@string{cacm="CACM"} +@string{jsac="J-SAC"} + +@string{conpar90vappIV="CONPAR 90 - VAPP IV"} + +@string{asplos91="ASPLOS"} + +@string{ipdps09="IPDPS"} diff --git a/correctness-model/writeup/confstrs-long.bib b/correctness-model/writeup/confstrs-long.bib new file mode 100644 index 0000000..55ff861 --- /dev/null +++ b/correctness-model/writeup/confstrs-long.bib @@ -0,0 +1,180 @@ +@string{mspc12="Proceedings of the 2012 ACM SIGPLAN Workshop on Memory Systems Performance and Correctness"} +@string{ppdp11="Proceedings of the 13th International ACM SIGPLAN Symposium on Principles and Practices of Declarative Programming"} +@string{padd88="Proceedings of the 1988 ACM SIGPLAN and SIGOPS Workshop on Parallel and Distributed Debugging"} +@string{spin04="Proceedings of the 11th International {SPIN} Workshop on Model Checking Software"} +@string{spin07="Proceedings of the 14th International {SPIN} Workshop on Model Checking Software"} +@string{spin08="Proceedings of the 15th International {SPIN} Workshop on Model Checking Software"} +@string{spin09="Proceedings of the 16th International {SPIN} Workshop on Model Checking Software"} +@string{fcad10="Proceedings of the Conference on Formal Methods in Computer-Aided Design"} +@string{jlap="Journal of Logic and Algebraic Programming"} +@string{iwops99="Proceedings of the 8th International Workshop on Persistent Object Systems"} + +@string{ijpp="International Journal on Parallel Programming"} + +@string{sc06="Supercomputing"} +@string{podc96="Proceedings of the Fifteenth Annual ACM Symposium on Principles of Distributed Computing"} +@string{popl73="Proceedings of the Symposium on Principles of Programming Languages"} +@string{popl93="Proceedings of the Symposium on Principles of Programming Languages"} +@string{popl96="Proceedings of the Symposium on Principles of Programming Languages"} +@string{popl97="Proceedings of the Symposium on Principles of Programming Languages"} +@string{popl99="Proceedings of the Symposium on Principles of Programming Languages"} +@string{popl05="Proceedings of the Symposium on Principles of Programming Languages"} +@string{popl07="Proceedings of the Symposium on Principles of Programming Languages"} +@string{popl08="Proceedings of the Symposium on Principles of Programming Languages"} +@string{popl11="Proceedings of the Symposium on Principles of Programming Languages"} +@string{popl13="Proceedings of the Symposium on Principles of Programming Languages"} + +@string{dac07="Proceedings of the 44th Annual Design Automation Conference"} + +@string{issta11="Proceedings of the 2011 International Symposium on Software Testing and Analysis"} + +@string{sosp03="Proceedings of the Nineteenth ACM Symposium on Operating Systems Principles"} +@string{sosp05="Proceedings of the Twentieth ACM Symposium on Operating Systems Principles"} +@string{oopsla02="Proceeding of the 17th ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} +@string{oopsla03="Proceeding of the 18th ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} +@string{oopsla04="Proceeding of the 19th ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} +@string{oopsla05="Proceeding of the 20th ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} +@string{oopsla06="Proceeding of the 21st ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} +@string{oopsla07="Proceeding of the 22nd ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} +@string{oopsla08="Proceeding of the 23rd ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} +@string{oopsla09="Proceeding of the 24th ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} +@string{oopsla10="Proceeding of the 25th ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} +@string{oopsla13="Proceeding of the 28th ACM SIGPLAN Conference on Object-Oriented Programming, Systems, Languages, and Applications"} + +@string{scool05="Synchronization and Concurrency in Object-Oriented Languages"} + +@string{hotpar09="First USENIX Workshop on Hot Topics in Parallelism"} +@string{hotpar10="Second USENIX Workshop on Hot Topics in Parallelism"} + +@string{atc="Proceedings of the USENIX Annual Technical Conference"} + +@string{ismm08="Proceedings of the 2008 International Symposium on Memory Management"} +@string{ismm09="Proceedings of the 2009 International Symposium on Memory Management"} +@string{ismm10="Proceedings of the 2010 International Symposium on Memory Management"} +@string{esop09="Proceedings of the 2009 European Symposium on Programming"} +@string{paste08="Proceedings of Program Analysis for Software Tools and Engineering"} + +@string{ecoop10="Proceedings of the 24th European Conference on Object-Oriented Programming"} + +@string{osdi04="Proceedings of the 6th Symposium on Operating Systems Design and Implementation"} +@string{osdi08="Proceedings of the 8th Symposium on Operating Systems Design and Implementation"} + +@string{toplas02="ACM Transactions on Programming Languages and Systems"} +@string{toplas09="ACM Transactions on Programming Languages and Systems"} + +@string{sc01="Proceedings of the ACM/IEEE Conference on Supercomputing"} +@string{sc06="Proceedings of the ACM/IEEE Conference on Supercomputing"} + +@string{ics95="Proceedings of the 9th International Conference on Supercomputing"} + +@string{icse00="Proceedings of the 22nd International Conference on Software Engineering"} +@string{icse05="Proceedings of the 27th International Conference on Software Engineering"} +@string{icse10="Proceedings of the 32nd International Conference on Software Engineering"} + +@string{pact01="Proceedings of the 10th International Conference on Parallel Architectures and Compilation Techniques"} +@string{pact08="Proceedings of the 17th International Conference on Parallel Architectures and Compilation Techniques"} + +@string{cc08="Proceedings of the 17th International Conference on Compiler Construction"} +@string{cc10="Proceedings of the 19th International Conference on Compiler Construction"} +@string{cc11="Proceedings of the 20th International Conference on Compiler Construction"} + +@string{asplos91="Proceedings of the Fourth International Conference on Architectural Support for Programming Languages and Operating Systems"} +@string{asplos02="10th International Conference on Architectural Support for Programming Languages and Operating Systems"} +@string{asplos09="14th International Conference on Architectural Support for Programming Languages and Operating Systems"} + +@string{vee05="Proceedings of the First ACM SIGPLAN/SIGOPS International Conference on Virtual Execution Environments"} +@string{vee08="Proceedings of the Fourth ACM SIGPLAN/SIGOPS International Conference on Virtual Execution Environments"} + +@string{lcpc08="Proceedings of the 21st Annual Workshop on Languages and Compilers for Parallel Computing"} +@string{lcpc10="Proceedings of the Fourteenth Annual Workshop on Languages and Compilers for Parallel Computing"} + +@string{pldi="Proceedings of Programming Language Design and Implementation"} +@string{pldi93="Proceedings of the 1993 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi94="Proceedings of the 1994 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi95="Proceedings of the 1995 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi96="Proceedings of the 1996 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi97="Proceedings of the 1997 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi98="Proceedings of the 1998 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi99="Proceedings of the 1999 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi00="Proceedings of the 2000 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi01="Proceedings of the 2001 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi02="Proceedings of the 2002 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi03="Proceedings of the 2003 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi04="Proceedings of the 2004 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi05="Proceedings of the 2005 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi06="Proceedings of the 2006 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi07="Proceedings of the 2007 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi08="Proceedings of the 2008 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi09="Proceedings of the 2009 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi10="Proceedings of the 2010 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi11="Proceedings of the 2011 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi12="Proceedings of the 2012 ACM SIGPLAN Conference on Programming Language Design and Implementation"} +@string{pldi13="Proceedings of the 2013 ACM SIGPLAN Conference on Programming Language Design and Implementation"} + +@string{isca90="Proceedings of the 17th Annual International Symposium on Computer Architecture"} +@string{isca09="Proceedings of the 36th Annual International Symposium on Computer Architecture"} +@string{isca10="Proceedings of the 37th Annual International Symposium on Computer Architecture"} + +@string{ppopp97="Proceedings of the 6th ACM SIGPLAN Symposium on Principles and Practice of Parallel Programming"} +@string{ppopp07="Proceedings of the 12th ACM SIGPLAN Symposium on Principles and Practice of Parallel Programming"} +@string{ppopp01="Proceedings of the 8th ACM SIGPLAN Symposium on Principles and Practice of Parallel Programming"} +@string{ppopp09="Proceedings of the 14th ACM SIGPLAN Symposium on Principles and Practice of Parallel Programming"} +@string{ppopp10="Proceedings of the 15th ACM SIGPLAN Symposium on Principles and Practice of Parallel Programming"} +@string{ppopp11="Proceedings of the 16th ACM SIGPLAN Symposium on Principles and Practice of Parallel Programming"} +@string{ppopp12="Proceedings of the 17th ACM SIGPLAN Symposium on Principles and Practice of Parallel Programming"} +@string{ppopp13="Proceedings of the 18th ACM SIGPLAN Symposium on Principles and Practice of Parallel Programming"} + +@string{ispass09="IEEE International Symposium on Performance Analysis of Systems and Software"} + +@string{iiswc08="Proceedings of the IEEE International Symposium on Workload Characterization"} +@string{iiswc09="Proceedings of the IEEE International Symposium on Workload Characterization"} + +@string{iccl88="Proceedings of the 1988 International Conference on Computer Languages"} + +@string{micro05="38th Annual IEEE/ACM International Symposium on Microarchitecture"} +@string{micro10="43rd Annual IEEE/ACM International Symposium on Microarchitecture"} + +@string{hpdc07="Proceedings of the ACM International Symposium on High Performance Distributed Computing"} + +@string{cav07="Proceedings of the 19th International Conference on Computer Aided Verification"} +@string{cav06="Proceedings of the 18th International Conference on Computer Aided Verification"} +@string{cav08="Proceedings of the 20th International Conference on Computer Aided Verification"} +@string{cav04="Proceedings of the 16th International Conference on Computer Aided Verification"} + +@string{tacas11="Proceedings of the 17th International Conference on Tools and Algorithms for the Construction and Analysis of Systems"} + +@string{procomet90="Proceedings of the International Conference on Programming Concepts and Methods"} + +@string{dexa04="Proceedings of the 15th International Conference on Database and Expert Systems Applications"} + +@string{lics="Symposium on Logic in Computer Science"} +@string{lncs="Lecture Notes in Computer Science"} +@string{lcpc="Proceedings of Languages and Compilers for Parallel Computing"} +@string{vmcai="Proceedings of Verification, Model Checking, and Abstract Interpretation"} +@string{vmcai09="Proceedings of the 2009 Conference on Verification, Model Checking, and Abstract Interpretation"} + + +@string{ieeecmpscieng="IEEE Computing in Science and Engineering"} +@string{higherordersymbcomp="Higher-Order and Symbolic Computation"} +@string{ibmjrd="IBM Journal of Research and Development"} +@string{ieeetc="IEEE Transactions on Computers"} +@string{cacm="Communications of the ACM"} +@string{jsac="IEEE Journal on Selected Areas in Communications"} +@string{hpca07="Proceedings of the International Symposium on High Performance Computer Architecture"} +@string{conpar90vappIV="Proceedings of the Joint International Conference on Vector and Parallel Processing"} +@string{tocs="ACM Transactions on Computing Systems"} +@string{toplas="ACM Transactions on Programming Languages and Systems"}, +@string{spaa98="Proceedings of the Tenth Annual ACM Symposium on Parallel Algorithms and Architectures"} +@string{tpds="IEEE Transactions on Parallel and Distributed Systems"} + +@string{jvm01="Proceedings of the 2001 Symposium on JavaTM Virtual Machine Research and Technology"} +@string{ics99="Proceedings of the International Conference on Supercomputing"} +@string{sc97="Proceedings of the 1997 ACM/IEEE Conference on Supercomputing"} +@string{sc01="Proceedings of the SC2001"} + +@string{cgo10="Proceedings of the 8th Annual IEEE/ACM International Symposium on Code Generation and Optimization"} +@string{cgo11="Proceedings of the 9th Annual IEEE/ACM International Symposium on Code Generation and Optimization"} + +@string{asplos91="Proceedings of the Fourth International Conference on Architectural Support for Programming Languages and Operating Systems"} + +@string{ipdps09="Proceedings of the 2009 IEEE International Symposium on Parallel \& Distributed Processing"} diff --git a/correctness-model/writeup/evaluation.tex b/correctness-model/writeup/evaluation.tex new file mode 100644 index 0000000..1bf7c00 --- /dev/null +++ b/correctness-model/writeup/evaluation.tex @@ -0,0 +1,304 @@ +\mysection{Evaluation}\label{sec:evaluation} + +We have implemented \TOOL. Our evaluation focuses on the following +questions: (1) How expressive is \TOOL for specifying +the correctness properties of real-world concurrent data structures? (2) How easy is it +to use \TOOL? (3) What +is the performance of \TOOL? (4) How effective was \TOOL +in finding bugs? + +In order to evaluate \TOOL, we have gathered a contention +free lock, two types of concurrent queues, and a work stealing +deque~\cite{ppoppworkstealing}. As C/C++11 is relatively new +there are no C/C++11 implementations for many concurrent data +structures, thus we ported several data structures. The Linux +kernel's reader-writer spinlock and the Michael Scott queue were originally ported +for the \cdschecker benchmark suite. We also ported an RCU +implementation and Cliff Click's hashtable from its Java +implementation~\cite{clickhashtable}. We report execution times on an +Intel Core i7 3770. + +\mysubsection{Expressiveness}\label{sec:expressiveness} + +In this section, we evaluate the expressiveness of \TOOL by reporting +our experiences writing specifications for a range of concurrent data +structures. + +\mypara{\bf Lockfree hashtable:} We ported Cliff Click's hashtable, +which supports simultaneous lookups and updates by multiple threads as +well as concurrent table resizing. The implementation uses an array +of atomic variables to store the key/value slots, and uses +acquire/release synchronization to establish the synchronization +between hashtable accesses. + +Hashtable updates consist of two CAS operations --- one to claim +the key slot and one to update the value. When a \code{put} method +invocation successfully updates both the key and value, the +update is visible to other threads. Thus, both CAS operations are +ordering points for the \code{put} method, and we annotate both of +them as potential ordering points. The \code{get} method is ordered after +an invocation of the \code{put} only if it sees both the key and +value updates. Thus we annotate an ordering point for the +key read only if the key is null. We also +annotate an ordering point for the value read if +it reads the value slot. The test driver has two threads both of +which update and read the value for the same key. + +\mypara{\bf RCU:} As discussed in the example, this is a +synchronization mechanism used in the Linux kernel that allows concurrent reads +and updates. We ran this benchmark with four threads, two update the data +structure and two read the data structure. + +\mypara{\bf Chase-Lev Deque:} This is a bug-fixed version of a published C11 adaptation +of the Chase-Lev deque~\cite{ppoppworkstealing}. +It maintains a top and bottom index to a shared array of +references. In terms of synchronization, +when pushing an item into the sequential deque, +we attach a unique ID tag to that element. When stealing or taking an item, we +use that tag as the ID of the method call. Thus, we have (push, steal) or +(push, take) pairs that have the same method call ID. +In our test driver, one thread +pushes 3 items and takes 2 items while the other one steals 1 +item. + +\mypara{\bf Linux Reader-Writer Lock:} A reader-writer lock allows either +multiple concurrent readers or one exclusive writer. We can abstract +it with a boolean \code{writer\_lock} representing +whether the writer lock is held and an integer \code{reader\_cnt} +representing the number of threads that are reading. +We test this benchmark with a single lock that +protects shared variables. We have two threads that read and write +the shared variables under the protection of a read lock and a write +lock. + +\mypara{\bf MCS Lock:} This benchmark is an implementation of +the Mellor-Crummey and Scott lock~\cite{mcs-lock, + mcs-lock-url}. This lock queues waiting threads in a FIFO. +Our test driver utilizes two +threads that read and write shared variables with the protection of +the lock. + +\mypara{\bf M\&S Queue:} This benchmark is an adaptation of the +Michael and Scott lock free queue~\cite{lockfreequeue} to the C/C++ +memory model. We ran with two threads, one of which enqueues and the +other of which dequeues an item. + +\mypara{\bf SPSC Queue:} This is a lock-free single-producer, +single-consumer queue. +We used a test driver that has two +threads --- one enqueues a value and the other dequeues +it. + +\mypara{\bf MPMC Queue:} This is a multiple-producer, multiple-consumer +queue. Producers call \code{write\_prepare} to obtain a free +slot, update the slot, and call \code{write\_publish} to +publish it. Consumers call \code{read\_fetch} to obtain +a valid slot, read the slot, and call \code{read\_consume} +to free it. The specification focuses on the synchronization +properties which require \code{write\_publish} to synchronize with +\code{read\_fetch} to ensure the data integrity and +\code{read\_consume} to synchronize with \code{write\_prepare} to +ensure that slots are not prematurely recycled. The test driver +contains two threads, each of which enqueue and dequeue +an item. + +\subsection{Ease of Use} In addition to expressiveness, it is also important for +specification languages to be easy to use. In our experience using \TOOL +to specify the real-world data +structures in our benchmark set, we found that \TOOL was easy to use. \TOOL specifications have only three +parts --- equivalent sequential data structures, ordering points, and +synchronization properties, and we explain the reasons as follows. (1) Specifying +sequential data structures is easy and straightforward, and developers can often +just use an off-the-shelf implementation from a library. +(2) When developers specify ordering points, they +only need to know what operations order methods without needing to specify +the subtle reasoning about the corner cases involving +interleavings and reorderings introduced by relaxed memory models. Take the +Michael \& Scott queue as an example, we can easily order enqueuers with the +point when \code{enqueue} loads the \code{tail} pointer right before inserting the new +node. +(3) For synchronization, the fact that we allow specifying +synchronization at the abstraction of methods makes it easy. For example, we only used +36 lines to specify synchronization in a total 1,033 lines of +code (omitting blanks and comments). + +\begin{figure}[!htb] +\vspace{-.2cm} +{\footnotesize + \centering + + \resizebox{\columnwidth}{!}{ + \begin{tabular}{lrrr} + \hline + Benchmark & \# Executions & \# Feasible & Total Time (s) \\ + \hline + Chase-Lev Deque & 1,365 & 232 & 0.15 \\ + SPSC Queue & 19 & 15 & 0.01 \\ + RCU & 1269 & 756 & 0.11 \\ + Lockfree Hashtable & 30,941 & 25,731 & 11.39 \\ + MCS Lock & 19,501 & 13,546 & 2.62 \\ + MPMC Queue & 170,220 & 93,224 & 45.63 \\ % 1r2w + M\&S Queue & 168 & 114 & 0.05 \\ + Linux RW Lock & 148,053 & 405 & 13.06 \\ + % Results from: 2014-03-25-13:26 + % Flags: -y -m 2 -u 3 + \hline + \end{tabular} + } +\vspace{-.1cm} + \caption{\label{fig:benchmark_results} Benchmark results} +\vspace{-.3cm} +} +\end{figure} + +\mysubsection{Performance}\label{sec:performance} + +Figure~\ref{fig:benchmark_results} presents performance results for +\TOOL on our benchmark set. We list the number of +the total executions that \cdschecker has explored, the number of the feasible +executions that we checked the specification for, and the time the benchmark +took to complete. All of our benchmarks complete within one minute +and most take less than 3 seconds to complete. + +\mysubsection{Finding Bugs}\label{sec:findbugs} + +\begin{figure}[!htb] +\vspace{-.2cm} +{\footnotesize + \centering + + \resizebox{\columnwidth}{!}{ + \begin{tabular}{l|ccccc|c} + \hline + Benchmark & \# Injection & \# DR & \# UL & \# Correctness & \# Sync & Rate\\ + \hline + Chase-Lev Deque & 10 & 0 & 2 & 3 & 2 & 70\%\\ + SPSC Queue & 2 & 2 & 0 & 0 & 0 & 100\%\\ + RCU & 3 & 0 & 0 & 1 & 2 & 100\%\\ + Lockfree Hashtable & 5 & 0 & 0 & 0 & 2 & 40\%\\ + MCS Lock & 4 & 0 & 0 & 0 & 4 & 100\%\\ + MPMC Queue & 6 & 0 & 0 & 0 & 2 & 33\%\\ + M\&S Queue & 11 & 0 & 6 & 3 & 0 & 82\%\\ + Linux RW Lock & 8 & 0 & 0 & 0 & 8 & 100\%\\ + \hline + Total & 49 & 2 & 8 & 7 & 20 & 76\%\\ + \hline + + % Results from: 2014-03-25-17:08 + % Flags: -y -m 2 -u 3 + \end{tabular} + } +\vspace{-.2cm} + \caption{\label{fig:injection_results} Bug injection detection results} +\vspace{-.2cm} +} +\end{figure} + +The next component of the evaluation examines the effectiveness of \TOOL for finding bugs. + +\mypara{\bf New Bugs:} In the M\&S queue benchmark used +in~\cite{cdschecker}, the dequeue interface does not differentiate +between dequeuing the integer zero and returning that no item is +available, and it passed our initial specification. However, after +modifying the dequeue interface to match that in the original paper, +\TOOL is able to find a new bug that \cdschecker did not find. The +original test driver for this benchmark performed the enqueues first +to make it easy to write assertions that are valid for all executions. +\TOOL allows specification assertions to capture the behavior of the specific execution +and thus is able to discover the given bug. + +\mypara {\bf Injected Bugs:} To further evaluate \TOOL, +we injected bugs in our benchmarks by weakening the ordering +parameter of the atomic operations. These include +changing \code{release}, \code{acquire}, \code{acq\_rel} and +\code{seq\_cst} to \code{relaxed}. We weakened one operation per +each trial, and covered all of the atomic operations that our +tests exercise. While this injection strategy may not +reproduce all types of errors that developers make, it does +simulate errors that are caused by misunderstanding the +complicated semantics of relaxed memory models. + +This fault injection strategy will introduce one of two types of bugs. +The first type is a specification-independent bug, which can be +detected by the underlying \cdschecker infrastructure which includes +internal data races and uninitialized loads. The second type is a +specification-dependent bug, which passes the built-in checks +but violates the \TOOL specification. These include +failed assertions and synchronization violations. We +classifying bugs as follows. If \cdschecker reports a data +race or an uninitialized load, \TOOL reports the error and stops. +If not, \TOOL continues to check the +execution against the specification. It first +checks for violations of the preconditions and postconditions and then for +violations of the synchronization specification. + +Figure~\ref{fig:injection_results} shows the results of the injection +detection. The column \textit{DR} represents data races, \textit{UL} represents +uninitialized loads, \textit{correctness} represents a failed precondition or +postcondition, and +\textit{sync} represents a synchronization violation. The detection rate is the +number of injections for which we detected a bug divided by the total +number of injections. + +\mypara{\bf Linux Reader-Writer Lock:} Our initial specification for this +benchmark did not allow \code{write\_trylock} to spuriously fail. +However, when we checked this benchmark against that specification, +\TOOL checker reports a correctness violation. We then analyzed the code +and found that \code{write\_trylock} first subtracts a bias from the +\code{lock} variable to attempt to acquire the lock, and restores that +bias if the attempt to acquire the lock fails. In the scenario where +two \code{write\_trylock} are racing for the lock before the lock is +released, one \code{write\_trylock} can first decrement the lock +variable, the lock can be released by the original holder, and then +the second \code{write\_trylock} can attempt to acquire the lock. +Even though the history indicates that the lock is unlocked, it +still holds a transient value due to the partially completed first +\code{write\_trylock} invocation. Thus, the second +\code{write\_trylock} invocation will also fail. As the second +\code{write\_trylock} serializes after both the first unsuccessful +\code{write\_trylock} and the \code{unlock} operation, +the sequential specification would force it to succeed. We then modified +the specification +of \code{write\_trylock} to allow spurious failures so that our correctness +model fits this data structure. This shows +\TOOL can help developers iteratively refine the +specifications of their data structures. By analyzing the \TOOL +diagnostic report, developers can better understand +any inconsistencies between the specification and the +implementation. + +\mypara{\bf MCS Lock:} Three of the weakened +operations are not detected because they cause the execution +to fail to terminate (and hit a trace bound). We +reviewed the code and found that weakening any of those three +operations makes the lock spin forever. + +\mypara{\bf M\&S Queue:} Our test driver does not cause +an enqueue or dequeue thread to help another +enqueue thread update the tail pointer, which corresponds to two of the +undetected injections. + +\mypara{\bf Lockfree Hashtable:} Our experiment only focuses on the two +primitive methods, \code{get} and \code{put} without triggering the +\code{resize}. We were able to successfully check all executions from a test driver for the lockfree hash table that generates executions that have no program order preserving sequential histories. + +\mypara{\bf MPMC Queue:} The undetected injections in this benchmark are +primarily due to the limitation of our test driver. One +synchronization property of this benchmark is +that \code{read\_consume} should synchronize with the +next \code{write\_prepare} to ensure that a slot cannot be reused +before the consumer has finished with the slot. Our test driver is +unable to reach this case so those injections are not detected. + +From our experiments on concurrent data structures, we can see that +\TOOL checker can help detect incorrect memory orderings, +help developers refine data structure specifications, +and help determine whether strong memory orderings are really +necessary. Since \TOOL checker is a unit testing tool, it is limited +to small-scale tests to explore common usage scenarios of the data +structures. As a unit testing tool, \TOOL was able to find 100\% of +injections for many data structures and to find 76\% of the injections +on average. For our 49 injections, 10 of them were detected by checks +in \cdschecker, and 27 additional injections were detected by +\TOOL. This shows that by writing specifications, we detect +significantly more fault injections. diff --git a/correctness-model/writeup/example.tex b/correctness-model/writeup/example.tex new file mode 100644 index 0000000..3ee4f3d --- /dev/null +++ b/correctness-model/writeup/example.tex @@ -0,0 +1,165 @@ +\mysection{Motivating Example} +\label{sec:example} + +We will use the read-copy update (RCU)~\cite{rcu} implementation in +Figure~\ref{fig:rcuexample} as a running example. RCU data structures +are useful for usage scenarios with large numbers of reads and +relatively few updates. RCU based data structures scale to large +numbers of readers because read operations do not contain any +low-level store operations and thus do not generate coherence traffic to invalidate cache lines. +Lines~\ref{line:rcuBegin} through~\ref{line:rcuEnd} contain +the code for the RCU implementation, and +Lines~\ref{line:testcaseBegin} through~\ref{line:testcaseEnd} show a +test case for the implementation. + +The RCU implementation maintains a shared pointer +(Line~\ref{line:rcuSharedData}) +\code{node} to reference the data shared by readers and +updaters. A reader loads the shared pointer (Line~\ref{line:rcuReadLoad}) to read +the shared data, while an updater allocates a new struct +(Line~\ref{line:rcuUpdateAlloc}), loads the shared pointer +(Line~\ref{line:rcuUpdateLoad}), and then attempts a compare and swap (CAS) operation to update the shared pointer. This process repeats until the CAS is successful. + +\begin{figure}[h] +\vspace{-.2cm} +\begin{lstlisting}[xleftmargin=6.0ex] +#define mo_acquire memory_order_acquire +#define mo_acq_rel memory_order_acq_rel +struct Node {/*@ \label{line:rcuBegin} @*/ + int data ; + int version; +}; + +class RCU { + atomic node;/*@ \label{line:rcuSharedData} @*/ + public: + RCU() { + Node *n = new Node; + n->data= 0; + n->version = 0; + atomic_init(&node, n); + } + void read(int *data, int *version) { + Node *res = node.load(mo_acquire);/*@ \label{line:rcuReadLoad} @*/ + *data = res->data; + *version = res->version; + } + void update(int data) { + bool succ = false; + Node *newNode = new Node;/*@ \label{line:rcuUpdateAlloc} @*/ + Node *prev = node.load(mo_acquire);/*@ \label{line:rcuUpdateLoad} @*/ + do { + newNode->data = data; + newNode->version = prev->version + 1; + succ = node.compare_exchange_strong(prev,/*@ \label{line:rcuUpdateCAS} @*/ + newNode, mo_acq_rel, mo_acquire);/*@ \label{line:rcuUpdateCASMO} @*/ + } while (!succ); + } +};/*@ \label{line:rcuEnd} @*/ + +RCU x, y; // Define two RCU objects/*@ \label{line:testcaseBegin} @*/ +int r1, r2, v1, v2; +void thrd_1() { // Thread 1/*@ \label{line:thrd1} @*/ + x.update(1); + y.read(&r1, &v1); // r1 == 0 +} +void thrd_2() { // Thread 2/*@ \label{line:thrd2} @*/ + y.update(1); + x.read(&r2, &v2); // r2 == 0 +}/*@ \label{line:testcaseEnd} @*/ +\end{lstlisting} +\vspace{-.2cm} +\caption{\label{fig:rcuexample}C++11 read-copy update example} +\vspace{-.3cm} +\end{figure} + +\begin{figure}[t] + \centering +\vspace{-.3cm} + \resizebox{\columnwidth}{!}{ + \includegraphics[scale=0.33]{figures/rcuhistory} + } + \vspace{-.4cm} + \caption{\label{fig:rcuhistory}History of a possible execution of two RCU objects} +\vspace{-.4cm} +\end{figure} + +This implementation uses low-level atomic operations provided by +C/C++ to implement the RCU mechanism. For example, in +Line~\ref{line:rcuUpdateCAS}, the \code{update} method uses a CAS +operation to update the \code{node}. The memory +order parameters attached to the atomic operations ensure that memory +accesses that were performed before the \code{update} operation must +occur before memory accesses that were performed after the \code{read} +operation. + +\mysubsection{Consistency Model} +\label{sec:exampleMotivation} + +In the example code, two threads, Thread~1 and Thread~2, each update +one of the RCU objects \code{x} and \code{y}, respectively, and then +access the object updated by the other thread. Under the C/C++ +memory model, this example admits an execution in which both Thread~1 +and Thread~2 read the initial value 0. + +Figure~\ref{fig:rcuhistory} shows a possible order for this +execution in which the model checker processes statements in the order shown in +the +execution labeled Original Trace\footnote{Note that the C/C++ memory +model does not include any notion of a trace. We merely use the term +trace to refer to the order in which \TOOL prints the execution.}, +which is an interleaving of the invocation and response events of +API methods indicated by \lq\lq\_i\rq\rq and \lq\lq\_r\rq\rq, +respectively. + +This trace is rather counter-intuitive because it orders +Thread~1's \code{update} event before Thread~2's \code{read} event, +yet Thread~2's \code{read} event does not observe the updated value. +However, under the C/C++11 memory model, this is allowed since a load +is allowed to read a value that is written by an old store. + +It is important to note that it is impossible to produce a sequential history +for this example that preserves program order and produces the observed +behavior. This means that traditional notions of correctness that relate +concurrent executions to sequential executions that were developed for +the SC memory model cannot directly be applied to the C/C++11 memory +model. + +This example leads to the following conclusion --- if we want to +define the behavior of concurrent executions by relating them to +equivalent sequential histories, then we must give up on preserving +program order. + +We note that for this example, if we give up on program order, we +produce the sequential history in Figure~\ref{fig:rcuhistory} +labeled Reordered Trace. The data structure operations in this +sequential history have behavior that is consistent with the observed +behaviors in the example execution. + +\mysubsection{Synchronization Properties} + +A correct RCU data structure ensures the property that the invocation +event of an \code{update} operation happens-before the response event +of the subsequent +\code{read} operation. Informally, all operations that appear before the \code{update} operation must appear to occur before the operations that appear after the \code{read} operation. While these +semantics follow naturally for SC, for relaxed memory models we must +ensure that neither the compiler nor the processor performs +reorderings that violate these semantics. Therefore, when a reader +reads the fields of the +\code{node}, +data races are eliminated since the \code{read} operation synchronizes with the \code{update} operation. + + +In the C/C++11 memory model, the transitive closure of +\textit{sequence-before} and \textit{synchronizes-with} relations form the +\textit{happens-before} relation in the absence of operations with the \code{consume} memory order. Since every time an \code{update} operation finishes, a CAS operation to +\code{node} with release semantics is executed, and the subsequent \code{read} +loads the \code{node} reference with acquire semantics, the \code{read} operation synchronizes +with the \code{update} operation, providing the necessary synchronization properties. +Some RCU implementations may use consume semantics instead of acquire semantics, +however we do not use consume for the following two reasons: (1) most existing +C/C++ compilers do not support consume semantics and thus acquire does not +involve extra overhead and (2) using consume semantics provides a more complex interface to +clients that may require them to explicitly declare dependences throughout the code. + + diff --git a/correctness-model/writeup/figures/CVS/Entries b/correctness-model/writeup/figures/CVS/Entries new file mode 100644 index 0000000..998018a --- /dev/null +++ b/correctness-model/writeup/figures/CVS/Entries @@ -0,0 +1,42 @@ +/fence_sw.dot/1.1/Wed Jan 14 22:53:32 2015// +/fence_sw2.dot/1.1/Wed Jan 14 22:53:32 2015// +/fence_sw_r_collapse.dot/1.1/Wed Jan 14 22:53:32 2015// +/fence_sw_r_collapse2.dot/1.1/Wed Jan 14 22:53:32 2015// +/fence_sw_w_collapse.dot/1.1/Wed Jan 14 22:53:32 2015// +/fence_sw_w_collapse2.dot/1.1/Wed Jan 14 22:53:32 2015// +/ff_sc_rf.dot/1.1/Wed Jan 14 22:53:32 2015// +/ff_sc_rf2.dot/1.1/Wed Jan 14 22:53:32 2015// +/fw_sc_rf.dot/1.1/Wed Jan 14 22:53:32 2015// +/fw_sc_rf2.dot/1.1/Wed Jan 14 22:53:32 2015// +/lockhistory.pdf/1.1/Wed Jan 14 22:53:32 2015/-kb/ +/release_seq.pdf/1.1/Wed Jan 14 22:53:32 2015/-kb/ +/rmw_atomicity.dot/1.1/Wed Jan 14 22:53:32 2015// +/rmw_atomicity2.dot/1.1/Wed Jan 14 22:53:32 2015// +/rmw_mo.dot/1.1/Wed Jan 14 22:53:32 2015// +/rmw_mo2.dot/1.1/Wed Jan 14 22:53:32 2015// +/rr_mo.dot/1.1/Wed Jan 14 22:53:32 2015// +/rr_mo2.dot/1.1/Wed Jan 14 22:53:32 2015// +/rw_mo.dot/1.1/Wed Jan 14 22:53:32 2015// +/rw_mo2.dot/1.1/Wed Jan 14 22:53:33 2015// +/sc_mo.dot/1.1/Wed Jan 14 22:53:33 2015// +/sc_mo2.dot/1.1/Wed Jan 14 22:53:33 2015// +/sc_wr_mo.dot/1.1/Wed Jan 14 22:53:33 2015// +/sc_wr_mo2.dot/1.1/Wed Jan 14 22:53:33 2015// +/specdesign.pdf/1.1/Wed Jan 14 22:53:33 2015/-kb/ +/specworkflow.pdf/1.1/Wed Jan 14 22:53:33 2015/-kb/ +/wf_sc_rf.dot/1.1/Wed Jan 14 22:53:33 2015// +/wf_sc_rf2.dot/1.1/Wed Jan 14 22:53:33 2015// +/workflow.pdf/1.1/Wed Jan 14 22:53:33 2015/-kb/ +/wr_mo.dot/1.1/Wed Jan 14 22:53:33 2015// +/wr_mo2.dot/1.1/Wed Jan 14 22:53:33 2015// +/ww_mo.dot/1.1/Wed Jan 14 22:53:33 2015// +/ww_mo2.dot/1.1/Wed Jan 14 22:53:33 2015// +/ww_sc_fence_first_collapse.dot/1.1/Wed Jan 14 22:53:33 2015// +/ww_sc_fence_first_collapse2.dot/1.1/Wed Jan 14 22:53:33 2015// +/ww_sc_fence_mo.dot/1.1/Wed Jan 14 22:53:33 2015// +/ww_sc_fence_mo2.dot/1.1/Wed Jan 14 22:53:33 2015// +/ww_sc_fence_second_collapse.dot/1.1/Wed Jan 14 22:53:33 2015// +/ww_sc_fence_second_collapse2.dot/1.1/Wed Jan 14 22:53:33 2015// +/rcuhistory.odg/1.2/Wed Jan 21 22:50:28 2015// +/rcuhistory.pdf/1.4/Wed Jan 21 22:50:15 2015/-kb/ +D diff --git a/correctness-model/writeup/figures/CVS/Repository b/correctness-model/writeup/figures/CVS/Repository new file mode 100644 index 0000000..aadc881 --- /dev/null +++ b/correctness-model/writeup/figures/CVS/Repository @@ -0,0 +1 @@ +Papers/15/issta.cdsspec/figures diff --git a/correctness-model/writeup/figures/CVS/Root b/correctness-model/writeup/figures/CVS/Root new file mode 100644 index 0000000..cd8b2f1 --- /dev/null +++ b/correctness-model/writeup/figures/CVS/Root @@ -0,0 +1 @@ +:ext:peizhaoo@plrg.eecs.uci.edu:/home/cvs/Papers diff --git a/correctness-model/writeup/figures/fence_sw.dot b/correctness-model/writeup/figures/fence_sw.dot new file mode 100644 index 0000000..1d4bbd4 --- /dev/null +++ b/correctness-model/writeup/figures/fence_sw.dot @@ -0,0 +1,20 @@ +digraph fence_sw { +margin=0 +ranksep=.3; + +N1 [label="A: fence(release)", shape=none]; +N2 [label="X: v.store(1)", shape=none]; +N3 [label="Y: v.store(2)", shape=none]; +N4 [label="Z: v.load()", shape=none]; +N5 [label="B: fence(acquire)", shape=none]; +N6 [style=invis, shape=point]; + +//N1 -> N5 [label="sw", color=blue, penwidth=3, constraint=false]; +N2 -> N3 [label="hrs", color="#555555"]; +N3 -> N4 [label="rf", color=red]; +N1 -> N2 [label="sb", color=blue, weight=100]; +N4 -> N5 [label="sb", color=blue, weight=100]; +N2 -> N6 [style=invis]; +N6 -> N3 [style=invis]; +{rank=same; N3 N4 N6} +} diff --git a/correctness-model/writeup/figures/fence_sw2.dot b/correctness-model/writeup/figures/fence_sw2.dot new file mode 100644 index 0000000..ed4ec19 --- /dev/null +++ b/correctness-model/writeup/figures/fence_sw2.dot @@ -0,0 +1,14 @@ +digraph fence_sw2 { +margin=0 +//ranksep=.3; + +N1 [label="A: fence(release)", shape=none]; +N2 [style=invis, shape=point]; +N3 [style=invis, shape=point]; +N4 [label="B: fence(acquire)", shape=none]; + +N1 -> N4 [label="sw", color=blue, penwidth=3, constraint=false]; +N1 -> N2 [color=invis, weight=100]; +N3 -> N4 [color=invis, weight=100]; +{rank=same; N2 N3} +} diff --git a/correctness-model/writeup/figures/fence_sw_r_collapse.dot b/correctness-model/writeup/figures/fence_sw_r_collapse.dot new file mode 100644 index 0000000..ee169e1 --- /dev/null +++ b/correctness-model/writeup/figures/fence_sw_r_collapse.dot @@ -0,0 +1,18 @@ +digraph fence_sw_r_collpase { +margin=0 +ranksep=.3; + +N1 [label="A: fence(release)", shape=none]; +N2 [label="X: v.store(1)", shape=none]; +N3 [label="Y: v.store(2)", shape=none]; +N5 [label="B: load(acquire)", shape=none]; +N6 [style=invis, shape=point]; + +//N1 -> N5 [label="sw", color=blue, penwidth=3, constraint=false]; +N2 -> N3 [label="hrs", color="#555555"]; +N3 -> N5 [label="rf", color=red]; +N1 -> N2 [label="sb", color=blue, weight=100]; +N2 -> N6 [style=invis]; +N6 -> N3 [style=invis]; +{rank=same; N3 N5 N6} +} diff --git a/correctness-model/writeup/figures/fence_sw_r_collapse2.dot b/correctness-model/writeup/figures/fence_sw_r_collapse2.dot new file mode 100644 index 0000000..6cc0981 --- /dev/null +++ b/correctness-model/writeup/figures/fence_sw_r_collapse2.dot @@ -0,0 +1,14 @@ +digraph fence_sw_r_collpase2 { +margin=0 +//ranksep=.3; + +N1 [label="A: fence(release)", shape=none]; +N2 [style=invis, shape=point]; +N3 [style=invis, shape=point]; +N4 [label="B: load(acquire)", shape=none]; + +N1 -> N4 [label="sw", color=blue, penwidth=3, constraint=false]; +N1 -> N2 [color=invis, weight=100]; +N3 -> N4 [color=invis, weight=100]; +{rank=same; N2 N3} +} diff --git a/correctness-model/writeup/figures/fence_sw_w_collapse.dot b/correctness-model/writeup/figures/fence_sw_w_collapse.dot new file mode 100644 index 0000000..3c0a864 --- /dev/null +++ b/correctness-model/writeup/figures/fence_sw_w_collapse.dot @@ -0,0 +1,18 @@ +digraph fence_sw_w_collapse { +margin=0 +ranksep=.3; + +N1 [label="A: store(release)", shape=none]; +N3 [label="Y: v.store(2)", shape=none]; +N4 [label="Z: v.load()", shape=none]; +N5 [label="B: fence(acquire)", shape=none]; +N6 [style=invis, shape=point]; + +//N1 -> N5 [label="sw", color=blue, penwidth=3, constraint=false]; +N1 -> N3 [label="rs", color="#555555"]; +N3 -> N4 [label="rf", color=red]; +N4 -> N5 [label="sb", color=blue, weight=100]; +N1 -> N6 [style=invis]; +N6 -> N3 [style=invis]; +{rank=same; N3 N4 N6} +} diff --git a/correctness-model/writeup/figures/fence_sw_w_collapse2.dot b/correctness-model/writeup/figures/fence_sw_w_collapse2.dot new file mode 100644 index 0000000..f1325f5 --- /dev/null +++ b/correctness-model/writeup/figures/fence_sw_w_collapse2.dot @@ -0,0 +1,14 @@ +digraph fence_sw_w_collapse2 { +margin=0 +//ranksep=.3; + +N1 [label="A: store(release)", shape=none]; +N2 [style=invis, shape=point]; +N3 [style=invis, shape=point]; +N4 [label="B: fence(acquire)", shape=none]; + +N1 -> N4 [label="sw", color=blue, penwidth=3, constraint=false]; +N1 -> N2 [color=invis, weight=100]; +N3 -> N4 [color=invis, weight=100]; +{rank=same; N2 N3} +} diff --git a/correctness-model/writeup/figures/ff_sc_rf.dot b/correctness-model/writeup/figures/ff_sc_rf.dot new file mode 100644 index 0000000..981e3fe --- /dev/null +++ b/correctness-model/writeup/figures/ff_sc_rf.dot @@ -0,0 +1,18 @@ +digraph ff_sc_rf { +margin=0 +ranksep=.3; + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="X: fence(seq-cst)", shape=none]; +N3 [label="Y: fence(seq-cst)", shape=none]; +N4 [label="B: v.store(2)", shape=none]; +N5 [label="C: v.load()", shape=none]; + +//N1 -> N4 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +N2 -> N3 [label="sc", color=purple]; +N1 -> N2 [label="sb", color=blue, weight=100]; +N3 -> N5 [label="sb", color=blue, weight=100]; +N4 -> N5 [label="rf", color=red]; +N2 -> N4 [style=invis]; +{rank=same; N4 N5} +} diff --git a/correctness-model/writeup/figures/ff_sc_rf2.dot b/correctness-model/writeup/figures/ff_sc_rf2.dot new file mode 100644 index 0000000..40f84ce --- /dev/null +++ b/correctness-model/writeup/figures/ff_sc_rf2.dot @@ -0,0 +1,18 @@ +digraph ff_sc_rf2 { +margin=0 +//ranksep=.3; + +N1 [label="A: v.store(1)", shape=none]; +//N2 [label="X: fence(seq-cst)", shape=none]; +//N3 [label="Y: fence(seq-cst)", shape=none]; +N4 [label="B: v.store(2)", shape=none]; +//N5 [label="C: v.load()", shape=none]; + +N1 -> N4 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +//N2 -> N3 [label="sc", color=purple]; +//N1 -> N2 [label="sb", color=blue, weight=100]; +//N3 -> N5 [label="sb", color=blue, weight=100]; +//N4 -> N5 [label="rf", color=red, constraint=false]; +//N2 -> N4 [style=invis]; +//{rank=same; N2 N3} +} diff --git a/correctness-model/writeup/figures/fw_sc_rf.dot b/correctness-model/writeup/figures/fw_sc_rf.dot new file mode 100644 index 0000000..68ead64 --- /dev/null +++ b/correctness-model/writeup/figures/fw_sc_rf.dot @@ -0,0 +1,16 @@ +digraph fw_sc_rf { +margin=0 +ranksep=.3; + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="X: fence(seq-cst)", shape=none]; +N3 [label="B: v.store(2)", shape=none]; +N4 [label="Y: v.load(seq-cst)", shape=none]; + +//N1 -> N3 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +N1 -> N2 [label="sb", color=blue]; +N2 -> N4 [label="sc", color=purple]; +N2 -> N3 [style=invis]; +N3 -> N4 [label="rf", color=red]; +{rank=same; N3 N4} +} diff --git a/correctness-model/writeup/figures/fw_sc_rf2.dot b/correctness-model/writeup/figures/fw_sc_rf2.dot new file mode 100644 index 0000000..6f947b1 --- /dev/null +++ b/correctness-model/writeup/figures/fw_sc_rf2.dot @@ -0,0 +1,16 @@ +digraph fw_sc_rf2 { +margin=0 +//ranksep=.3; + +N1 [label="A: v.store(1)", shape=none]; +//N2 [style=invis, shape=dot]; +N3 [label="B: v.store(2)", shape=none]; +//N4 [style=invis, shape=dot]; + +N1 -> N3 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +//N1 -> N2 [label="sb", color=blue]; +//N2 -> N4 [label="sc", color=purple]; +//N2 -> N3 [style=invis]; +//N3 -> N4 [label="rf", color=red]; +//{rank=same; N3 N4} +} diff --git a/correctness-model/writeup/figures/lockhistory.pdf b/correctness-model/writeup/figures/lockhistory.pdf new file mode 100644 index 0000000..3563fea Binary files /dev/null and b/correctness-model/writeup/figures/lockhistory.pdf differ diff --git a/correctness-model/writeup/figures/rcuhistory.odg b/correctness-model/writeup/figures/rcuhistory.odg new file mode 100644 index 0000000..4575354 Binary files /dev/null and b/correctness-model/writeup/figures/rcuhistory.odg differ diff --git a/correctness-model/writeup/figures/rcuhistory.pdf b/correctness-model/writeup/figures/rcuhistory.pdf new file mode 100644 index 0000000..07df89c Binary files /dev/null and b/correctness-model/writeup/figures/rcuhistory.pdf differ diff --git a/correctness-model/writeup/figures/release_seq.pdf b/correctness-model/writeup/figures/release_seq.pdf new file mode 100644 index 0000000..5a264d5 --- /dev/null +++ b/correctness-model/writeup/figures/release_seq.pdf @@ -0,0 +1,344 @@ +%PDF-1.5 %âãÏÓ +1 0 obj <>/OCGs[6 0 R]>>/Pages 3 0 R/Type/Catalog>> endobj 2 0 obj <>stream + + + + + Adobe Illustrator CS4 + 2012-11-08T19:47:19-08:00 + 2012-11-08T19:47:19-08:00 + 2012-11-08T19:47:19-08:00 + + + + 256 + 120 + JPEG + /9j/4AAQSkZJRgABAgEASABIAAD/7QAsUGhvdG9zaG9wIDMuMAA4QklNA+0AAAAAABAASAAAAAEA AQBIAAAAAQAB/+4ADkFkb2JlAGTAAAAAAf/bAIQABgQEBAUEBgUFBgkGBQYJCwgGBggLDAoKCwoK DBAMDAwMDAwQDA4PEA8ODBMTFBQTExwbGxscHx8fHx8fHx8fHwEHBwcNDA0YEBAYGhURFRofHx8f Hx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8fHx8f/8AAEQgAeAEAAwER AAIRAQMRAf/EAaIAAAAHAQEBAQEAAAAAAAAAAAQFAwIGAQAHCAkKCwEAAgIDAQEBAQEAAAAAAAAA AQACAwQFBgcICQoLEAACAQMDAgQCBgcDBAIGAnMBAgMRBAAFIRIxQVEGE2EicYEUMpGhBxWxQiPB UtHhMxZi8CRygvElQzRTkqKyY3PCNUQnk6OzNhdUZHTD0uIIJoMJChgZhJRFRqS0VtNVKBry4/PE 1OT0ZXWFlaW1xdXl9WZ2hpamtsbW5vY3R1dnd4eXp7fH1+f3OEhYaHiImKi4yNjo+Ck5SVlpeYmZ qbnJ2en5KjpKWmp6ipqqusra6voRAAICAQIDBQUEBQYECAMDbQEAAhEDBCESMUEFURNhIgZxgZEy obHwFMHR4SNCFVJicvEzJDRDghaSUyWiY7LCB3PSNeJEgxdUkwgJChgZJjZFGidkdFU38qOzwygp 0+PzhJSktMTU5PRldYWVpbXF1eX1RlZmdoaWprbG1ub2R1dnd4eXp7fH1+f3OEhYaHiImKi4yNjo +DlJWWl5iZmpucnZ6fkqOkpaanqKmqq6ytrq+v/aAAwDAQACEQMRAD8A9LXvmXQ7HXdN0G6u1i1b WEnk021IYmVbVVeajAFRxVwaMRXelaHFVlz5v8p2qSPc63YQJC4ilaS6hQJI0ssARizCjGa2ljof 2kZeqmiqAs/zM/Lu8svrtv5m0trURpM8hvIE4RyOI0aQMyslXYL8QHxGnXFVSL8wvJc+oWGn2urQ 3dxqbBLJrXlcROxSWQL60IeJSUtpD8TD7OKrta/MDyPoi3/6V16xtZNMUPf27zxmeLlE06KYFJl5 vEhZFC8mH2QcVVW87eTUleF9e09Jo/W9SJ7qFXT6tH60/JSwZfSiPOSo+FdzQYqvsfOHlXUNQj0/ T9Xs7y9ki9dILeeOZvT4o4b4C1OSSKy1+0u4qK4qm0jrGjOxAVAWYkgAAb9Ttirxz8r/AM/L7zl5 qk0a40uygilmeKD6netdTwCG0SaRpl9JEkh9XnGLiNghbiqhh+8ZV7JirTuqIzsaKoJY+AGKsd0X 8xvJGs6KNbstXhXSyzILm652Y5R2/wBcb4bpYW2tgZq0/u/j+zvird7+Y3kCw1E6be+Y9Ntr5PVE kEt1CjI0HH1Fkq1EZfUHwtQ/ccVXzfmD5EheSN/MOnGaKL13t0uoXm9P0xMGESMztWNg4oNwQR1x VdD548qyWGoahNqC2FnpM62upTakkmniCV1jZBJ9cWAqHE6cW6NyFCcVUG/Mr8vklkjk8xafGsSR u873MawfvnmjRROxETOXtJRwDchxNRiqtcef/IttFLLP5i0yKOBTJMzXkACqspgZj8ewEymP/WHH riqO8v8AmDRvMWjWmtaLdJe6Xep6ltcpUBlBKnZgrKQwIIIqDscVTDFXYq7FXYq7FXYq7FXYq7FX Yq7FXYq7FXYqwrzj+U3lvzZrUetajPdxahBAlravC8bRRxr64lH1eeOaCX10umjlEyOCoAAHxVVQ up/kt5avbrUbiK9vrBtUv01S7+rNb1FxEVaMwySwSywcJPVcGF1blLJU0ICqpQP+ccPJR09tOm1D VJrIxpEkLS268FQ2X2Wjt0f4o9KtozUn4VqKMzMVUVon5H2Og2DWuj61dxyQXLanpl7cpFcTxak9 pdWj3E7FVW4Qrd8/T4r8S1LHkcVVPPP5CeTfOd1e3Op3WowS39wbm5+qTRxqxNtb2vplWicMgFlE 4DVIcVB7Yqh3/wCcfPKbalcXy6rqy/WZ7y5ltGmt5bUvfxzRuDBNBLGyx/XJjGGBoXatQxGKofS/ yf8ALHknW7PVbfVtVuDcyNEVuHF1cGcW04WRJFVZFUJLcySKFartyHEA1lEkXIDiEdz12sDf4kMJ z4RbPbfRbe5hWa31i9mhb7MkdyGU022IGZEdaCLEYfJYkSFg2FT/AA5/2tNQ/wCR/wDzbh/N/wBC HyZUgPMHkj9L6DqWlfpe+i/SFrPa+o8nqqvrRtHyaP4OYHL7PIV8cv0vaPhZY5OCB4JA8q5G+fRB CRaNpGmflj5YdNb8w6prEFwEg5XhadEdUYcbdArGFWX9lpCNuuV9udtQzS8Q44YhZ+gbm/5x5E+d BpzZ44Y3JLvLX5C+QYfKUOmp9dl066exvQtz9VjuB9V5ssbvbwxNxlimeOVCSKM5HF3d218JcQB7 26E+KIPeiZvyH0KbVm1eTXtX/STTi7M6GwjX1xLb3HqiBLNYA7T2UMjt6dXK/FUVGSZJLqn/ADjn 5RsAmsaXc6qdQ05LZrOAPFcQ87K3itYJHtisXqmKOItxDqzcnA+0FwSNC2zFDilVge/l+Ps79k30 jynpM2manDfarqt9pOs3tpq7NcwWzyreWl2Ll3e5sofT9NzBDEY5BWNY+IIGy1DOKsgj8eVtOWRx yMckZRI8v0iwkifkZ+Wum6DILXzPq0NnJLYCS5gu7eZ5G055XtICBBIJQrSikfE/3aUHw72RmJcj axnGW4Nquif848eTOaapY3OuaZc26tFpi3MtuTAoooeOF4pQKwqsP7wciigsOfx5JkzH8sfIF75G 8tWPl5dWF/YWIlEYNuImb1ZWkqzc3+zzptSvXFWRX9jrE0/qWep/VUoB6LQJKtR3qSrb998VRF42 pR2oNlHFcXS0qkztCjD9r4lSUg+Hw4qt025vriJje2ZspVbjw9RJVYUHxKy70/1gD7YqusNT07UI jLY3MV1Gp4s0Tq9D4Gh2PscVROKuxV2KuxV2KuxV2KuxV2KuxV2KuxV2KuxV2KuxVJ/MpMKabfjf 6nf25KnoRck2TE/6q3Rb6MzNHvxx/nQP+x9f+9pBRdxo9pLM1xHztbpqFri3bgzEdOY3ST/Zqc18 sQJsbHy/G/xa5YYk2Nj3j8b/ABWi11mP4Y7+OROzT2/KT6TFJCv/AAuDhmOvzH6iEcMx/F8x+oh5 /rf5vWumebn8sXy34kiuYLSXUbK0thArz/UwGLT3Mz8UbVLcNSGvxbVoTjwzPM/IfrteCZ5y+Q/X ac+S9U8o+edIuL1LCeZQYo5RqojklKXNtDfQunF5kRJIbqNwqlaHYqCMB08DzHF790HTQP1Di9+/ 4+DNsub3Yq7FWPeYIbrTW/SeiRM+p3MixPYKCYbpyNjLTaJlVa+t4CjcvhGVmG9hzMeoBjwZd49D /FH3d4/on4UbXaUtlq9up1GJU1y2VUvgim3uInP8rI5cIf2WVyrDocjwxnzG/wBrjanQiJBNEHlI dfcefvGxHUIto9XsAGic6lbD7UMnFbkD/IccEeng9D/lHGpx5eofb+Pxbi1OHL1D7f2/H5sb/MXz b5s0/wAtwah5IsU1W/aeZLi3ltri4KJBZXNxwaKF4pEkeeCOEcujOBSpGWQmJCw2QmJCwk/lnzv+ aN15n0zStZ0JYbOSS4hv7yOxvIo2WGXUEFwk0kjxQLS0taRycmk9fkpouSZvUMVdiqFOlab9fXUP qsQvlBUXIUCTidiCw3I+eKoW9uPMFte+pBbRX2nPxDRRt6VzF2ZhzPpyjvSqn54qmmKuxV2KuxV2 KuxV2KuxV2KuxV2KuxV2KuxV2KoTV9OTUtKvNPkcxpdwyQGRd2X1FK8hXutajLcGU45xmP4SD8lK l5f1KTUtGtbuZQl0ycLyJdxHcxExzx/7CVGX6MnqsIx5DEfT084neJ+IooCYZjpS648teXLi7a8u NKs5rt3Ej3ElvE0hdTEVYuVLFgbWGh/4rT+VaKqmk6Jo2j2zWukWFtp1s7tK8FpCkCGRqcnKxhQW NBU4qjcVdirsVY1qPmLWW8m69rWkWYub60gvpNDtQrym5e2jYQgxpxdvVlTZVNSpFN8VSjyPYeb9 W0W6vvNlyYfMMepXyadcQWv1P6vaxTGGJEjcu0sEwi9X96zVDjf4VbIShfvcjBqTjsEcUDzieR/U e4j7rDKLHV5RdLp2qItvqBBMLLX0LhRuWhY96fajPxL7r8RYz3o82eXTDh48Z4ode+P9b9EuR8js u1C2mglOpWKFp1A+tW6/8fEajoO3qL+we/2TsaiE4kHijz6+f7e75OtyRIPFHn18/wBvd8kdbXEF zbx3EDiSGZQ8bjoVYVBy2MgRY5NsZCQsciqYWTsVdirsVS/VtGh1ERSCWS1vbeptbyBuMkZalRQ1 V1ag5KwIOKqzXkNlDapqFzGJ5mSBZSPTWWcrWiqS1OXE0FfbFUVirsVdirsVdirsVdirsVdirsVd irsVdirsVSOz/wBxvmS6sj8NrqoN7Z+AuEAS6jHYchwlUdSTIe2ZuT95hEv4oek+7+E/fHy9IR1T zMJLsVdirsVdiqnc3ENtby3E7cIYUaSVzU0VBVjt4AYq61jto7aKO2RY7dVAhjjUKoWmwVRQAUxV UxVQvrC0v7Zra7jEsLUNKkEMDVWVloysp3VlNQdxglEEUWzFmljlxRNH8fMd46vLPM/5o635U836 Zot5DLJoSThLzWbmFozLFIvEBXoEf0S3JnWnKlKDcthZNRKEgDy73r9B2Bi1mlnliQM3D6YRN0R5 cxxcgDy533HGgaP5w078xZYb69mu9DeC4urRl/dw83kXkrxoFj9QGQnpv1+VOHFljnokmFE+X9r5 vgw5YZ6JJhRPl8uV7vQs2jtXYq7FXYq7FVG9srS+tZbS7iWa2mXjLE4qCMVSvSl1TTrsaXcCS8sG Umxvz8Toq/7puD4gfYk/aGx+L7SqdYq7FXYq7FXYq7FXYq7FXYq7FXYq7FXYqluv6ZNf2A+qsI9R tHFzp0rEhVnjB4huO/B1LRyU6ozDMnS5hCXq+iW0vcf0jmPMBBCto+qQ6pp0V7Epj58llhf7cUsb FJYnp+1HIrK3uMhqMJxTMT/aDuD8RuoKMylLsVdirsVUNQtFvbC5s2PFbmJ4WbrQSKVrSo8cVYZ5 28k6vqs+kz2kNlqg06yurT6tqLrCkdzObdodQg52mpRevB9XYJzhI+M9qhlUm89fln5z8xarqUY1 ZZ/LOqNYCTSp7q5hWJEntnv+KxIwfnFp6JEoKbzTNVa/Gqx3Q/yl/OSzuTe6r5ij1O4kitYbqNNW 1S19WK3lleSL1kjeSP1jMsvqLuhUxqBG5oqj/Lnkj8xPLfka98kC3fWZLqK1GnalLfyfo6142lvF cRSSC4ttRjj9eGV0SCMrxdQKfEAkJjIxNjYsR1X8kvzcljsjq/mS0uZm0270e6mn1jUoVumvrdI4 YgiwcI0imjV2Vf8AejgOY+I0BIDKMJTugT1P62d/l5pF75f1+K8uTaSXOp2lvputVvI/rK3FpNqM 8l8yH7aXDSwqiA81D/EBwOQ8aF1Ycfx4XVi/exf/AJUx+daSae8nme31JNPMFxFFf6lqzE3P6Okt p29ZOMyD6xNI6GN0Yq1GNFRRY2qkH5T/APOQi29pH/jaOKRWtzdTG+1CdlMQPMxh0VZOTcnKuOLC TgaCGJsVZZ+T3kTzr5X8webLjzBKJLTVmsZrPjf3GogyxQtFOxluljm5fCtSy7jiBsuKvUcVdirs VSvT7nVE1K4sL+MyJ8U9lfIlEaIt/dSU2WSPkB/lLv2bFU0xV2KuxV2KuxV2KuxV2KuxV2KuxV2K uxVj1/8A7gdWfVl20jUGRdXH7ME4ASK89kKhY5j2AVtgrnNhi/f4/D/ykPp8xzMff1j8RuTFHJkO a9LsVdirsVdiqWaKl/DLqFtdmSSOO5eSznkJblDP+941O/7t2ZAPADFUzxV2KuxVTuLeC5geC4jW aCUFZInAZWU9QQdjgItlCZiQYmiElmW50eNkuFfUdCP2i4M09sv+WDyaaIeO7r35DcVkcPPeLmmM NTttHJ3coy/RGX+xPSuoqLRtIkgSbTj9VjkUPFLYyenGwIqr8UPpP7clODwY/wAO3u/FOrnpREkU YEd232frC4nWbPc01K3HYBYrkf8AEYpP+Ep74+uP9Ifb+o/Yx9cf6Q+R/UfsRdlf2t7EZbd+QU8X Ugq6MOqujAMrDwIrlkJiQsNkMgkLDzi9/wCcgfKNjqmoWF3puqRJpzXiTXxitzAxsRe8+HGcyH1D pVwqVQbr8XEHJM2deWvMFtr+l/pCCCa14z3NpNbXAQSxz2dw9tMjek8sZ4yRMKq5B8cVTTFXGtDT Y9id8VSzQtSubqKa2vkEep2L+ldqoIRqiqSx1r8Ei7jw3HbFUzxV2KuxV2KuxV2KuxV2KuxV2Kux V2KrZI45Y3ilQSRyAq6MAVZSKEEHqDhBINjmqQaZNJoV5Fol45bT524aHeOanZS31OVj1dFUmNj9 pNjVlLNn5ojPE5Y/WPrH+/Hkf4h0O/I0ByZDmvS7FXYq7FUv12/u9PsDewRCaO3dXvI6Eubcf3pj oftqPip3pTviqOilimiSWJg8Uih43U1DKwqCD7jFV2KuxV2Kpb5l1+z8veXtS129SSS00u2lu7iO EKZGSFC7BA7IvKg2qwxVjHkzWNU1jTLrX9N0ttNhN/e2z6JNPG/rC0uHt5J1aOscE7SRPVAzIxH2 t+eVmJG4+TmxzxyARy9OUuo8j3x+0dLHpZfpup2moQGW3Y1Rik0LgpLFINykiHdWFfu3GxGSjIFo z4JYzUvgehHeD1/HVSv9PkaT69Y8Y9RRaAsSI5lG/pzUrUfytSq9tqgwnD+KP1ff7/xs4eTHvxR+ r7/I/jZA2nl/yhqMTXf6FsjLK0n1hZbWEyCVmmEyyfCatyup+W5r6jnfmayhMSFs8cxIWE007TNN 0yyisdNtIbGxgBENrbRrFEgJLEKiBVWpJOwybNE4q7FUFrOotpumzX3omdLejzIpowiDD1HGxqUS rU70piqMR0kRXRgyOAysNwQdwRireKuxV2KuxV2KuxV2KuxV2KuxV2KuxVQv7C0v7OW0u4/UglFG XcEEGqsrChVlYBlYbg7jfLMWWUJCUeYVJtK1HUdOvU0TW3MzPUaXq5UBbpQK+lNxAVLlQNxsJB8S ftKmXnxQyR8XFt/Oj/N8x3w+2PI9DIMgzAS7FXYq7FUvudWjtNTtrGeIxw3alba6qOBmXf0T/KxX dfHcdt1UwxV2KuxVRvLOzvbSazvYI7m0uEMc9vMiyRyIwoyujAqykdQcVdaWdnZw+jaQR28PN5PS iRUXnK5kkbioA5O7szHuSTiqF1HSfXmW9tJfqmpRrxS5A5K6AkiOZKr6iVJ2qCKniVJyEoXuObk4 dRwjgkOLGend5xPQ/YeoLWnav687WV3F9U1ONeT2xPJXQGhkgei+oladgVqOQUnGM72PNObTcI44 nixnr3eUh0P2HoSgtS1TTdE1JL66uorex1AGO4aR1UCaJSUk3PdFKN7hBlOTJHHKyaEvv/G3ydXl mMU7kaEufv7/AJbfJObS6t7y1hu7ZxJb3EaywyCoDI4DK2/iDl8ZCQBHIuRGQkARyKrkmTsVcyqy lWAKkUIO4IOKoWwksED2FmVUaeEgaBaj0x6asi79uBGKorFXYq7FXYq7FXYq7FXYq7FXYq7FXYq7 FUPqGn2eoWctlexCa2mFJIzUdDUEEUKspAKsDUHcb5ZiyyxyEomiFSZNRvNAYW+syGfSSQttrTfa jqdo72gHH/Jn+yej8WoXzDijn3xisnWHf5w/4nmP4bH0jkyHNel2KuxVTntre4QR3ESSoGVwrqGA ZCGVqHurCoxVCjWLYasdLmVoLhkEls0lAk6gfH6RB3KftKd+/TfFUdirsVS3zLr9n5e8valr18sj 2el20t3cJCA0jJChdggYqpYgbVI+eKpH/wArR8q2+l3Ooaw82i/UruawvbW8j5zRXMFq16Y2+qm5 Q8rZPUQqx5VCj4yFxVK5/wA9vINtc3trdPew3VnfnTRAbOd3nkFz9T5wemrh0M6yIo+23pycVPHF Ut8zfmF5e83XK+T/AC5qL2mvXaxz6Rq1xZ3UdtIT9Yr9Uugq0cpaS8ZB8DrUIXPILVmgZRoc3Zdk 6vFgzCeUGWPrEdfeDsR5d9MDm1PQND8l6t5Q8+NcJrHlv1LvSZbS1uZpnt5ObM8QaOPnB6kThpXK xbgcwRmEdF4mPgnzHIs/a38pr8oy4SQZDcEUYyG3u3Fcj0eg+Xvzl8nRaba6bZRX18mn/o2wluor YxRFbwWsNtcqLhoX9CRrxODcasAzKCtCc7FDhgI9wDpsMOGAj3ABlHmH8y/JXl3WY9G1jUGtdQli juFj+r3MiCKVpFVmljjeJf8AeeVjyYUVGY/CpOWNiUWH55/lxc2tjPPe3GnPqFvb3cMF7Z3UTeld AmNuYjaFlorksjlaI7V4oxCq0/nt+XDajpNhbXlxdSaverpsUkdpcLFDcuXVY7h5UjEb84+PA/GK glePxYqzSDTRDq15frJUXcUEbQ0pRoTJV+Vd+SyKOn7OKozFXYq7FXYq7FXYq7FXYq7FXYq7FXYq 7FXYq0yq6lWAZWFGU7gg9jhBpUgGmaloR5aKn1vSur6MzBWhA/5YnYhVH/FLkJ/KyAUOd40M/wDe +mf8/v8A64/3w37xJHJNNM1fT9TieSzl5tE3p3ELApLE9K8JY2AdGpvRh036ZjZsE8ZqQ58u4+48 ioKMylLsVUri0tbgxG4hSUwuJYS6huEi9HWvQjxxVA3uttYX6xXlq6WMvFYdQSskYkJpwmVRyj36 Mar4kYqmeKoPWNI07WdJvNJ1KH6xp+oQyW13ByZOcUqlHXkhVlqp6qQcVSWP8tfJa6XNpclg11ZT 38eqzJd3FzdM17CyMkxknkkkrWFajlRtwQQTVVLH/JL8spLiK6k0h5LyEMI7t7y9a4DNdNemQzGb 1DJ9ZkaT1C3Kp64qqj8o/J9vIlxpS3WlahBJFLZ30FxJO9sYROqJBFeG6t44wl7OojEXAczQA0IV QUn5D/ltcz315qNjPf6pqgYarqT3dzBPdB+QcSC1e3iCvy+JERUO3w7Ciqin5V+VPL1/LdWemTSa LcJaC8tIp7qeRZNPlhls5WSR5ZHS2+qosaRMoVfh9NwfhhKRHucrDihkFXw5PP6T8f4T79j3jqaX P5ZeQfMNlp0+qQtrzW0FlHDqdxcyvLMLAS+hLI8TorvW6lLNT4ixr2pIG+TjzhKBMZCiEvtv+cff yhtmdodApzWJGVru9dSsI4x/C0xGyVT3Qsp+FmBLFdD+Q35ZWzaW1hpstj+h7qC9s0hurkx+tbuW UyRSSSRyEhijMy8uPw1oBirL7H6++uanLNzWzRbeC1RieDFFaSSRB7mYIT/k07YqmWKuxV2KuxV2 KuxV2KuxV2KuxV2KuxV2KuxVgv5o+UvOfmOLTovLOrPpLQ/WUvZlvLm1HCeH00ZY7ZOUskbHnHyk VVI3V67Kr/yr8qedfL+mXo84a42t6pdTh45BNNLDHFwDlEWULxInklAp/usIO1MVZJqnl+yvplu1 Z7PU4l4Q6jbEJOq1rwNQyyJX9iRWXvSuZOHVSgOH6oH+E8v2HzFFBCEXVtZ0v4Nbt/rNsOmrWMbs v/Pa2HqSx/6yF16klBtlpwY8u+I1L+bI/dLYH48J6DiW05tLy0vLdLm0njubaUVjniZXRh0qrKSD mJPHKB4ZAgjoUquQV2KpfqttrDmKbS7tIZYuXK2nTnBKDTZitJEIpsyn5qcVVn1CK0sY7nVJIrIk IJi0g9NZGoOIkYJX4tgaDFUSjq6h0IZWAKsDUEHoQcVbxV2KuxV2KsN8yeYvK/lzXLS2TWbTTtf1 eaKOHRJXLfXZLlzFG5gjDyR83Ur9YC8RT4+VKZWYVuHMhqRKIhlFxHI/xR93eP6J+BiyTTtXgvHe 3ZHtr6EVnspqCRQTQMKEq6Hs6kjt1BGSjO/e15tMYAS+qB5SHL9h8jv8FLXNWlso4oLSMXGqXhKW Vua8SRTlJIR0jjBqx+jqRknHTMVpv174q7FXYq7FXYq7FXYq7FXYq7FXYq7FXYq7FXYq7FXYq7FX YqlF55ZsZbiS8s5JdL1CU8pLuyYIXbpyliYPBMabAyRsR2pmXj1kgBGQE4DpLp7j9UfgQilL675n 0/a9s49Vtx1utP8A3UwFK/FazORQeKTMx7JkvDwZPpkYHuluP9MB98QP6S7ovT/MWi38/wBWt7oC 8AJNlMrQXIVdixt5gkoX3K0yrLpMkBZHp7xvH/TCx9q2mOY6WnRHQo6hkYUZSKgg9iDiqG0/StO0 6N4rC3S2ikcyNFGOKciADRRsvToMVQ1gPMkVwIr82l1bUP8ApcPqQSAjpWFvVU18RIPlird/5g07 T5/SvBPEKA+v9XmeHft6qIyA+xOKo6W6toYhNNKkURpSR2Crv03OKro5YpQTG6uBsSpB/VirD9R8 teTNS89af5gl1Mfp219BbW0SeCjfVY71FHplTIarqMpajfsr4GqqaeZbXUL14baxsQbpB6kGrSSC JLVjtVeJMrttugXiw2Y0yMogt2HUSx3XI8weR9/4sdFWHVLzTnS31wqUNEh1aNeMLk7BZl39Fyff gx6EE8cjxEbS+bkHTxyjixc+sOo/q/zh/sh1BA4k5qK0rudwPlljguxV2KuxV2KuxV2KuxV2KuxV 2KuxV2KuxV2KuxV2KuxV2KuxV2KobUNM03UYPq+oWsN3BXl6U6LItfGjA75ZizTxm4ExPkaWku/w 3Jb76Xql5ZU39F5PrkJNdqrdCZ1XtxjdMyPzYl9cIy+HCf8AY0PmCim+fm+3qGisNRXqZEeWyb/V EbC7Vj7mRca08us4fKX2+n7iu7h5jkiI+u6RqFop2D+kl0CflZPcuP8AZKMfygP0zhL48P8AuxFb bHnDyyu1xqEVk/8Avq9rZyf8i7gRP28MH5DN0iZf1fUPnG1sJhZ6lp18nOyuobpOvKGRZBTp1UnK MmGcNpAx94pNqk9tb3EZiuIkmjO5SRQy/ca5Wqyz0+wskaOztorZGPJkhRYwT0qQoGKqdoukx3M0 FoIEuYqG4ii4B157rzVdxyptXJGEgASNiqA0/wA5eXb+LVbiC6aO00SWaHUr25hmtbaN7VnS443E 6RxSLE0Th2RmVabnIqo3XnvyMmoPpF3rVjHcuFQwTyoqOZZprX0lZyI3k9a1mjaMEsCpBGKQSDYe Qw6y/lf8xdL8wS+ZtL1LQdXE1raWtleeqUsF9R2aKJmkpFA9uxqjEclKD4iFzCjhyRyXzH6Hsc/a mhz6A4aMMw9VkfVMecepG1kDz73sEn5geQ41LyeZNLRAHJZr23ApG3BzUv8Ast8LeBzNeNQ91+Z/ 5c2sFrcz+ZtMW1vXCW90LuFoSWExUtKrFEVvqsqhmIUspUHltiqZ6b5m8u6pe3FlpupW17dWgrcx W8qylPiZCGKEjkrIQy9R36jFUzxV2KuxV2KuxV2KuxV2KuxV2KuxV2KuxV2KuxV2KuxV2KuxV2Ku xV2KoC88v6Dev6l7ptrdPWvOaCOQ1NN6sp8Mvx6rLAVGUh7iQikP/g/ylv8A7hLDfrS2hH/GuWfn 9R/qk/8ATFaDv8JeXv8Alk/5KSf81Y/nsvf9gWgkY/KLySPNY80/V7gaqgQQsl1PGkfpigoI3QtX 9oOSD4Zm/wAu6nwPAscG/wDCDd+8fdSOEXanpH5R+WNN1PW74M9yuvm/+v280FgKjU5fVuFFzFbR XhUnYK87ClP5Vpp2SR/9C6eTTLBI2paq3oWMel8TLbEPZh3lnjdvq/Mm5mleWV+XPkfgZF+HFVsv /OPehLbWwttd1Vr20lW7S5vXguhNdxT3dxDNdAxRvLwk1KaqrInIH4qkVxVE6f8AkPothdapdw+Y 9ckudYurW+vpp5rSd2uNPnFzZuGltXI9CXdV+yRswYUxVJPK/wDzjP5dstI0JNY1C8m1LSnsLuSO CSE2oubF55isfO3WVoZJ7yV2DnlvsV6YqyvyF+UOjeSdWuL3TL66uLeWKRI4LtllkRpmi9T98ApZ FjtYUjUiqBT8Rrsq/wD/2Q== + + + + + + Adobe PDF library 9.00 + + + application/pdf + + + release_seq_final + + + + + 1 + False + False + + 558.000000 + 261.000000 + Points + + + + + TeXGyreHeros + TeXGyreHeros + Unknown + Version 2.037;PS 2.000;hotconv 1.0.51;makeotf.lib2.0.18671 + False + MyriadPro-Regular.otf + + + + + + Cyan + Magenta + Yellow + Black + + + + + + Default Swatch Group + 0 + + + + + + xmp.did:01801174072068118F62BC2FBF2D0DDE + uuid:b6550317-f02c-e943-99b1-a50cb06f5ab2 + xmp.did:F77F11740720681184ABF8CADCB1BF49 + proof:pdf + + uuid:ba4dbd99-36b2-2d4a-a0d2-e8c58b81b3c8 + xmp.did:01801174072068118F62D9F7CDB3ECB2 + xmp.did:F77F11740720681184ABF8CADCB1BF49 + proof:pdf + + + + + saved + xmp.iid:F77F11740720681184ABF8CADCB1BF49 + 2012-11-08T17:43:18-08:00 + Adobe Illustrator CS4 + / + + + saved + xmp.iid:F87F11740720681184ABF8CADCB1BF49 + 2012-11-08T18:19:13-08:00 + Adobe Illustrator CS4 + / + + + saved + xmp.iid:01801174072068118F62D9F7CDB3ECB2 + 2012-11-08T18:34:52-08:00 + Adobe Illustrator CS4 + / + + + saved + xmp.iid:01801174072068118F62BC2FBF2D0DDE + 2012-11-08T19:47:17-08:00 + Adobe Illustrator CS4 + / + + + + + + + + + + + + + + + + + + + + + + + + + + + + endstream endobj 3 0 obj <> endobj 8 0 obj <>/Resources<>/Font<>/ProcSet[/PDF/Text]/Properties<>>>/Thumb 12 0 R/TrimBox[0.0 0.0 558.0 261.0]/Type/Page>> endobj 9 0 obj <>stream +H‰ÌWÛnG }߯˜G莇œ{ßjÇPÀ(Ò胮Ç-ìNÚ¦Ÿßùh¥UKN ++‹æpx;‡ÜãŸNÕñù©Q'/NÕ@J>Þ Ç/_õîã`”QÞ'Åâ·Ãíð‘üáؤ¿¨?£½a…'ɃYýür`õI9u®.ˆßu£†G‹¤œÍšBðŠ’×^]?”›tÌ°Œoç¼Â?­µjÌ^3iòAdc‘_£×)fH¢ÓތD8IQtX§„³äY›VFpˆ’ÑQ §““@ØhŠP†]+Ñk\¢¬…;±á V1è`Þ:9í3— ®‡»áõð +Ÿ“e‰Æ”D.éR^Þä´1¦*÷¿ƒb‰%'§–Ã⇣å’Ê”¡³¼ßW!'U ª€‹üS.¸&ÐýHl‚U`c7úW05Áûv-wM骂·í–Ô¯ýuщ]BM%Û*¸X|w”áï›å"64·í綻Æ}3Ýݹ€ +›ÅU3•ÝÙÊT÷òHÙé'-TÚÊpœgØ~ƒ ó᦭ wGx–a$ø©îxºód#Í-Ü®ño¸6nZtÎmäýl¹›. + €#&J Ê%ð':ðQû Y¾/ЪŠ@ÕÚ©È:dxŽ׉„ 0 +Á=~_w ¯WÏ9hÊõkØ~Òd“Œ@œÓÁ5`ouÈoO'¨™'Iƒ'w™Ëe#nOÆ+'ì˜ÁsN!>J0 +¢òÂz-ÍêfhL¡tVí¶k=ÀӒ_yoJå^¹¤c@TlPɘ©$ã¡&#ùÞ¶BÁ…{q½'ò³ÞŠÏ ÃŽ‘9X/ÐñD¸Ê(¤ÙöÆ]ur?>¡°«\Ío¸n‚þû±Ûè>ýݶ ~ï˜ëaìÁÏBT¨EM¨Üªª±yLAt ±Ã¤C×0چbR–}U`ò(Dš:g=çH4e†£Ó0 ×0¤¨­V)@ +:ã²CZ‡Ô…pnÀ¨'f0~Ôlž¶|{xÛ¸Þ%úˆÌâ¶O¯í ÓOnQý¬.w2¼.§ F0·jƛ#ÊxT¹É[]כj>]¿Šûw Ú/pÿÖ°Þâ~{@§~ŽÚØYM›XÜ°m¹©v.¸8‚ײWÁ¢þ`<·TÌøŸS#–¼(¼¸7M[0RÊ9+&Ò!Áâ5õÑþþ‰`ª÷³@P  ¸k­£˜F$aqºßÛ[ö 6ž2ODºUáÛºîÄÂÔ¯âáEU ®×ýÐh½ +G’ÝÕ]»‘Ñ5î:1öîºlqµù]màäy×­6Ž'#û( ÝR›÷_ÅþW8±Vöò ‡|L(<¤î¡¬®è´€Mjd#öü„•Òdê2yp@.‰=Y‘ý`ët.¤Ž7-Ì '˒ÃÛÕ!Ôî ¼Àå5.„ð̀g¹Ì4þLr¢¬È  Ž(öǝ•%®¼§bÈùìâ&îzµowÖað‹ æ¥á'_¶RHou +Í×zj³LY +„иV&Ìn–°n°cÒ²æRºƒ*ÔG»óHd +_õ¦ÓRzô–¬ÆëÃ~¯ÕÙ$1‚ †\8,ø-/KX„²pëÁ]v~ªðõŸÒj‰ endstream endobj 12 0 obj <>stream +8;X^=acXLq#Xrl5!h:6uI-P+\M-NSacjlKJm&EpL/R(-:5Mo"@rfI.==)2c"%X:Ua-9Wt7C;u[ +:bOoR%n$'j&P3-EB-'m6Q?j0^Ak"Gfd%mUtM,X&9FB!22>F!O5]<*CYKrfdBFp\@$ +GZ%N63\rfIN*Se]%P1]:F^\4GQ;OqUCXmM'Wha.32p33moLj_\WI#0,K3QG[k2^I0q^H'>7amk"$WP12 +]<;58LL#1)~> endstream endobj 13 0 obj [/Indexed/DeviceRGB 255 14 0 R] endobj 14 0 obj <>stream +8;X]O>EqN@%''O_@%e@?J;%+8(9e>X=MR6S?i^YgA3=].HDXF.R$lIL@"pJ+EP(%0 +b]6ajmNZn*!='OQZeQ^Y*,=]?C.B+\Ulg9dhD*"iC[;*=3`oP1[!S^)?1)IZ4dup` +E1r!/,*0[*9.aFIR2&b-C#soRZ7Dl%MLY\.?d>Mn +6%Q2oYfNRF$$+ON<+]RUJmC0InDZ4OTs0S!saG>GGKUlQ*Q?45:CI&4J'_2j$XKrcYp0n+Xl_nU*O( +l[$6Nn+Z_Nq0]s7hs]`XX1nZ8&94a\~> endstream endobj 6 0 obj <> endobj 15 0 obj [/View/Design] endobj 16 0 obj <>>> endobj 5 0 obj <> endobj 17 0 obj <> endobj 18 0 obj <>stream +H‰|T Li®¢»ªaÔV(‹u»fªj=@Q :è ‚ˆ‚$ŒÝÒÜØ݀JëDéæÒu‚àȀ8Š‚0\‚ 2^Èpj{Ž'‡ºÊxìh^áÏl¶šÄd“M6ùóþÿ½ÿßûò½W…cR ÇñÉ˼6¬÷öŸ¤Úè½C£òQi´æ8'0¸0E*|9žFóQΧOŸüx9Q˜dÝöå/m0 1û7>Äš¨HµŽŸ¿h‘³£ÙºŒY7GÞyÞ¼ycօ_¦LWñ;´:Uœ–_‘ ILЄéTJ'~Yl,?¡å5*­J“lhtªñžßÁ;Šñu¼øæ[WGWwg'~µX9LÉGiyU”N­Òða"@d”XC#æé4aJU\˜&†O0ßü—»õÿÐá£ây‹Ž2{:1¨åÃâ•sE”„±* Iñ:M”Jë4wE`ЎDïÎ+U[1Qìpl†MÄ0c0lªóÄ°å¶ÇVãX†­“aSÍ⭝ï±zìö§ñüïø#‹µ?X|”|%q’KÇKWK›‰IÄZ"¸A<$ƒÈ<™Dæ-+’õ[αŒ´¼i5ݐÙ" ·à¢Þ"ɔ +†‘€Q 9è*–Â?ƑˆG—h0;£‡e£äwcgXŠDŸZió ™=9j£4aä2þëm lñ¥ýÜ|6Z¢ =hr‡;ÛCz½  +V(À·hÓös +̸-@èŽ'Ý09 6ZÂFo ý±ÞäMû6D!ò B´×LV>;/gÉ}dEÿÔԒҧ{§è—'òIþ–r=Øà­0Nzºªôä¹*m™Z­IŒŒ,M¬àä¹ÅÉBç£ÌÐ?¥ª@oût„õ$©^4õë" ډlRž›iŽ˜p$Â[jȖ]¼@ÀW0¥Æ+^¼Fü2ßø­›ÙŸV'N7Ÿ¾Æ —¯YéðµZŒœ¹>D°L¤a ÛÁ¿_H@¯èª’“•Uº²È(Ͷ­j3ôQú?19àÅɠ{"ÇB;uOð5} © ±;GÓÄíυÄÙjâXÝêeAmn]îϖÙd®27âèæn´~Šüçý=PÐ ºn›»ý ôyA}¸›M§ŸjL¿Î€ÅÝãõ\噒óWo=/­:ÊæÈ(¡ïDÕ.Å¥¤u[xԅñ_³ŸŠë_Kç”äÔ0×b–ø‡Ì㍬™~û÷­çúýèC%Ï3à7d‡dpÈ×~òŠÉÊIbåúL¤ß‚(~G찇0…F§M£øŸ …I&tÒ¡€”µøe‰§t“`h‘É“õÑɐÙ:m* ì)Õ,̇úuä;4sMÀNM{<€(©hÊodÞdå-W*öʐÖ´UÈJáÕ²â÷Þ֊ê“lI¥^C< œ5ÖÖ)6ÑTó÷{ {ÙðÝ᫘ͪ²«ÿ¼èµ‘“wéûTÉ’^ˆM±1õùAªÙÖ4•ºÅbŒfïTó¡âÌúU]Nìc‹g° ìÜÞ¢ K£‚c9#\q!ûG—ÓÙGròØ®c—ë®1¯ZÜg!I°§Óª uí»8CnF^žB¤¿)`}xõ D°Lë5Ñi›˜Ùßô‚ôíõÇ ¹p6M_Ìï&òµ¡ùä<áÈÍ|5ì:/å•pòvýMåg²Õƒpª_”Ç~$„FuFYuRH¥'ƒ¬ìk‡ìŸÏ«WkÛÊÆä8èFÜ'gÍõ +á‘Ìeô ½;#͐ÆR;»‚#˜9!ÿØöÛó® ¡Aù\Ž>+U¯õ½àÑS;ðê'÷T"„C2›•L1,Z€þæŠHd‡ìg‚ý¶¢‚rvW±;F©÷cœ=›aœ‘Ëê ÛW?bޔºepr)þ\»añMüQ¿–Ø +®`Ս¬Êdw~è¼SVt`>{¤‡8¼sÛá&|óž˜h.B¹kµ·BçãÓ=ê*“·Šù‹L¢¸iô“íœ' ³òå³ME 5W˜Ç§|—p¨|je°xZotì¾Ômlbj’n·ÆÒ—rô£ˆ²=å=à67w ^‚«­°}Áèvõ®ačŽNÒÄÅkÏT—ž.O*‰/N 1W5— Qà‰ˆ²øÖ^qržAœyQM¶Ï <š]xèÛPPÛÐÆtW-åPý3h•æ=˜ÏÖVW]dú*ƒæp¨ìTÉÀñÎ746-)žMIMIIM°4’W’7Õø2‹¿ÛšÈÄ!L@,%³#²#6*äú.Áª ¯‚Æ! ” ûhôWÞ¹¢9ïgƒ-X‚õ}ðû9ïÇWÒÔN_„p?7Wÿß>>?ÿú_œü:òJÿ>ïÛûy’†{B÷ $õ°%»>»†u'©aÃrÂH6j7Uy23•òEö/fqÿRuÓI.›„‰Ò=†é{Xÿ”õ:fVèãaÀ~¹÷¼«14¸ÀŽ^ŒZR €Íø +ÌÀà÷à³¢w­¼}À÷ÎÇáÓ_µ8ªL¤o³¯§©¿¸®zž Ëœq`˜Z$KK=Q´\ãkËs¤uabmlL_]›pû@€Ôcvø4ÎèápèøÒ1@HÖûœ°R•`£pt ¬}´­¦‘‘œ‡Î‡~}‰sZz­Ç÷™÷Îû÷»€ÔbvøxwÑã÷|ã¸ørøx3û¼zˆ{†c{bbP> endobj 10 0 obj <> endobj 19 0 obj <> endobj 20 0 obj <>stream +%!PS-Adobe-3.0 %%Creator: Adobe Illustrator(R) 14.0 %%AI8_CreatorVersion: 14.0.0 %%For: (Brian Demsky) () %%Title: (release_seq_edit3.pdf) %%CreationDate: 11/8/12 7:47 PM %%Canvassize: 16383 %%BoundingBox: 6 14 564 275 %%HiResBoundingBox: 6 14 564 275 %%DocumentProcessColors: Cyan Magenta Yellow Black %AI5_FileFormat 10.0 %AI12_BuildNumber: 367 %AI3_ColorUsage: Color %AI7_ImageSettings: 0 %%RGBProcessColor: 0 0 0 ([Registration]) %AI3_Cropmarks: 6 14 564 275 %AI3_TemplateBox: 487 171 487 171 %AI3_TileBox: -3 -211.5 573 522.5 %AI3_DocumentPreview: None %AI5_ArtSize: 14400 14400 %AI5_RulerUnits: 2 %AI9_ColorModel: 1 %AI5_ArtFlags: 0 0 0 1 0 0 1 0 0 %AI5_TargetResolution: 800 %AI5_NumLayers: 1 %AI9_OpenToView: -74 406.5 2 1835 980 18 0 0 43 129 0 0 0 1 1 0 1 1 0 %AI5_OpenViewLayers: 7 %%PageOrigin:181 -225 %AI7_GridSettings: 72 8 72 8 1 0 0.8 0.8 0.8 0.9 0.9 0.9 %AI9_Flatten: 1 %AI12_CMSettings: 00.MS %%EndComments endstream endobj 21 0 obj <>stream +%%BoundingBox: 6 14 564 275 %%HiResBoundingBox: 6 14 564 275 %AI7_Thumbnail: 128 60 8 %%BeginData: 5086 Hex Bytes %0000330000660000990000CC0033000033330033660033990033CC0033FF %0066000066330066660066990066CC0066FF009900009933009966009999 %0099CC0099FF00CC0000CC3300CC6600CC9900CCCC00CCFF00FF3300FF66 %00FF9900FFCC3300003300333300663300993300CC3300FF333300333333 %3333663333993333CC3333FF3366003366333366663366993366CC3366FF %3399003399333399663399993399CC3399FF33CC0033CC3333CC6633CC99 %33CCCC33CCFF33FF0033FF3333FF6633FF9933FFCC33FFFF660000660033 %6600666600996600CC6600FF6633006633336633666633996633CC6633FF %6666006666336666666666996666CC6666FF669900669933669966669999 %6699CC6699FF66CC0066CC3366CC6666CC9966CCCC66CCFF66FF0066FF33 %66FF6666FF9966FFCC66FFFF9900009900339900669900999900CC9900FF %9933009933339933669933999933CC9933FF996600996633996666996699 %9966CC9966FF9999009999339999669999999999CC9999FF99CC0099CC33 %99CC6699CC9999CCCC99CCFF99FF0099FF3399FF6699FF9999FFCC99FFFF %CC0000CC0033CC0066CC0099CC00CCCC00FFCC3300CC3333CC3366CC3399 %CC33CCCC33FFCC6600CC6633CC6666CC6699CC66CCCC66FFCC9900CC9933 %CC9966CC9999CC99CCCC99FFCCCC00CCCC33CCCC66CCCC99CCCCCCCCCCFF %CCFF00CCFF33CCFF66CCFF99CCFFCCCCFFFFFF0033FF0066FF0099FF00CC %FF3300FF3333FF3366FF3399FF33CCFF33FFFF6600FF6633FF6666FF6699 %FF66CCFF66FFFF9900FF9933FF9966FF9999FF99CCFF99FFFFCC00FFCC33 %FFCC66FFCC99FFCCCCFFCCFFFFFF33FFFF66FFFF99FFFFCC110000001100 %000011111111220000002200000022222222440000004400000044444444 %550000005500000055555555770000007700000077777777880000008800 %000088888888AA000000AA000000AAAAAAAABB000000BB000000BBBBBBBB %DD000000DD000000DDDDDDDDEE000000EE000000EEEEEEEE0000000000FF %00FF0000FFFFFF0000FF00FFFFFF00FFFFFF %524C45FDA0FFA8A8FD7EFF27277DFD5EFFA827FFA8A87D7D52FD057D27A8 %FF7D7D527DA87DA87D7D7DFD05FF272752FD08FFA8F87DA8A87D7D52FD05 %7D527DFF7D7D52FD067D27FD3CFF7D277D7D277D27272752F8F8527DA8FF %2727F85227275227F87DA9A9A9FFA9A97EFD04A9FF554F7FFFA827527D27 %7DF8272752F8F8527D7DFF2727277DF827F8275227A8FD3BFFA87D7DA8A8 %FD087DA87DA8A8FD047D52A87D7D76FFFD0BA955F97FFFFF7DA8A8A8FD06 %7D587D7D77A8A8FD087D277DA8FFFFA87DA8A8FFA8FD4BFFAEAEFD0FFF83 %FD0BFFA1AEAEFD15FF7DA8A8FD48FF7E8383FD0DFF5E1083FD0AFFCA9AAE %83AE83AEA8A2A2FD0FFFA8FFFFFF7DA8A8FD43FFA9CBFF83AEAEFD07FFAE %AE83AE82FD0DFFC3FFFFFFAEAE83FD15FFA8A8FD43FFA3A9FFFFAEAE83AE %83AE83FD04AEFD11FFC3CAA8FFFFFF83ADA8FFFFFFA87D7DA87DFD0EFFA8 %A8FD40FFA3FD05FFAEFFFFFF527DA8A8A8FD11FF76C3FD05FFAE83FFFF7D %27272752A8FD0DFFA8A8A8FD3EFFA9A9FD08FFA8272752277DFD10FFA8AF %CAC3A8FD05FF83AEA8A87D7D7DFD13FFA8FD3CFF7FFD09FF7DA8A87DA8FD %13FFCAC3CAFD05FF83FD18FFA8A8FD3BFFA9A9FD1EFFA8277DFFC3C3FD05 %FF83FD17FFA8A8A8FD3AFFA9A9FD1BFFA8A8FF5227A8FFFFCAC3FD05FF83 %FD54FF7EFD1BFF7DFFFFA87DFD04FFCAC3FD04FFAE83FD19FFA8A8FD38FF %A9A9FD25FFC3FD04FFAE83AEFD18FF7DA8FD38FF7FFD26FFA1CABCFF5E10 %33FD19FF7DA8FD37FF7FA9FD26FFB5FCC3FF335EFD1AFF7DA8FD37FF7EFD %18FFA8A8FFA8FD09FFCAC38CB5FFAF83FFFFFFA8FD17FFA8FD37FFA9A9FD %17FFA8FFFF7D7DFF7DA852A852A87D52A8FF9BA15277527D52FFFFFD047D %A87DA87D7D27FD44FFA97FFD17FFA8A8A852A8A8272752F82727522752FF %59F852F827527D7DFF2727F827F827F8F827F8A8FD0DFFA8A8FD35FFA9A9 %FD19FFA87DFFA8A87DA87DA87DA87D7D52FD057DA87DA8A87D7D52FD057D %52FD0FFF7DFD36FF7EFD25FFA8FFFFFFCACA84AE83AEAEFD19FFA8FD36FF %7FFD29FFC3CAAFFFA8AEA7AEA8FD4DFFA97FFD15FF7DA8FD12FFCAC3CBFF %FFFFAEAE83AEAEFD15FFA8FD35FF7FA9FD15FF7DFD13FFCAC3CAFD05FFAE %AE83FD04FFA8527DA87DFD0BFFA8A8FD34FFA97FFD16FFA8FD13FFCAC3C3 %FD07FF83AEFFFF5227275227A8FD0AFFA8A8FD35FFA9A3FD2BFFCAC3FD07 %FF83AEFFA87DA87D7DFD43FF7FA9FD14FFA8FD16FFCAC3FD07FF83AEFD49 %FF7FA9FD14FF7DFD13FFA85284FFC3CBFD06FF83AEFD12FF7DFD36FF7FA9 %FD13FFA8A8FD12FF52F8CAFFFFA1CAFD06FF83AEFD11FF84A8FD36FF7FFD %27FFA8A8FD04FFCACAFD06FF83FD11FFA8FD38FF55FD2DFFCAC3CAFD04FF %AEA7FD4AFF55FD13FFA8A8FD19FFC3CAFD04FFAEFD4BFF78FD13FF7DA8FD %19FFC3A1FFCAFF5E5E83FD0FFFA8A8FD39FF7FA9FD12FFA8FD1AFFCABDB6 %FF5E1083FD10FF7DFD3AFF7FA9FD2CFFCAFC8CC3AF5D83FD10FFA8A8FD3A %FF7FA9FD12FFA8FD0DFF7D7DFFFFFFA87DFFA8FF7DA8C3BC93FF7DA8A8A8 %7DFFFFFFA8A8FD05FFA8A8FD3EFFA37FA9FD10FFA87DFD0CFF52527D7D52 %522727275252F8FFA84BFD04275252FF7D2727275227522752F852FD3FFF %A97FA9FD10FFA8A8FD0BFF52527DA8527D5227527D52527DA80552275227 %A852FF7D52275227522727522752FFA8FD3EFFA97F7FFD1DFFA8FFFFFFA8 %FFFFFFA8FFA87D7DFFA8FFA8A1A1FF7DFFFFFFA8FD05FFA8A8FFA8A8FD3F %FFA97FFD11FF7DA8FD1AFFCACACAFD0DFF7DFD42FF7FA3A9FD0FFF7DA8FD %1BFFC3C3FD0BFFA8FD3DFFA87DA87DFFFFFFA9A3A9FD2CFFCAC3FD48FF27 %27F852FD04FFA9A97FFD11FF7DA8A8FD19FFC3CAFD08FFA8FD3DFF7D7D7D %A8FD07FF7FA9A9FD0FFFA8FD1BFFA1CAFD06FF7DFD4AFFA97F7EFD11FFA8 %A8A8FD14FFA8277DCAC3CAFFFFFF7DFD4EFF7FA9A9FD0FFFA8A8A8FD13FF %5227A8FFFFCAC4FD52FFA97F7FA9FD12FF7DA8A8FD0EFFA87DFD04FFA19A %FD54FF7FA3A9FD11FFA8A8FFFFA8FD0FFFA8A8A8FFC3CAFD53FFA9A97F7F %7EFD12FFA87DA8A8FFA8A8A8FFFFFFA8A87DFFFFFFA8FFFFFFA1C3FFFFCA %FD53FFCB7FA9A3FD15FFA8A8A8FFFFFFA8FD09FFCACAB6C3FD55FFA9A97F %A97FA9A9FD21FFBCFC8CFD5AFFA9A97FA9A3A9A9FD15FFA8FD07FFC3B5C3 %FD5CFFA9FFA3A97EA97FA97EA9A9FFA9FFA9FFA9FFFFFFA9A9A9FF5252FF %A87D7D527D7DAF27767DA87D7D52A852527D7DA8FD58FFA9FFA9A97FA9A9 %A97FA9A9A9A3A9252BA97D27A87D52FD042752275252F87DFD0427F82727 %A8FD61FFA9FFFFFFA9A9A9FF7D7DA8FFFD047D52A8527D527D7D7D277D7D %A87D52A8FD79FFA8FD86FFFF %%EndData endstream endobj 22 0 obj <>stream +H‰ÜWýrÛ8¿Ð;ðæ&7ÉÌÆ´üíô/ç«ëÝ&öÄN¯{™Ž‡–`›JT)*©ûô’’¬$MânÛIöš©E?€€äÎ?ǓýA(ç°ß¬Õ‰·³s¤€i©ˆ¥’¡Yª•!í^ì¿U«#h0ìÍrà{P)—ñeY橑Þ=TœÅä¢ôz½Gv÷1åZ²`)ÌRø4ƒëf- {ÅÚ¨ì˜iÄù>íQ¿Aº­.Ÿ>‹oXšò/†ÛiöšH;”Yòxy(?ZAڝitÛÈû•_@úàXY±+@šI!Uz@ŽÖhú["‡‘?@yK ®+"§2֝‡·k¿‚’i…{BøUÌ`؞r¦ˆiâÛ  †~cv˜qžgÑ0€ÍNא›3kÔeŠÖ aflÈÝÙ0BÊ´F×p ø‹·‡UGhÿv¯.`Éí&bl?îåj•L"¦®ÓûA1Ì)D‰ÀM°1kõºÄïúÅ7G –»ß$û ߯µI»Û$íF£–ëØÄn8ܐsƒs ôÄmb«U¯»_ǹȨ˘›°5 ©ïü?“!ėò§‚Y·íŸ¿ùu€)SKиùRdÚfg¯XãûŽ­Ál³ï%Oå{kã~·EZõzÓ ~¯Ù&ý*îYý­&ñýrI¿øuz££PŽ»·3Æ-)¾äñßó1N¶Û¼·Š‡›½ë6HÏýXj½Êÿ~ñßيnk qn;æÌÑY%êµ³ .{‡G22±7)yˆ›c^¹t¼rl9(ž%ޕ×ìÓO™Ô¢.¤ß¡KÅn]îÑAÈA!'¥…lz`Ù +Áè 2 ô\câ0^–ÌA˜ÕFYÀ¦ÅBÀgÊ6'Ϭò PVÒ£àD¡" +¥(wê¹Ãð +†—˜Øª÷¨tXé°²‚•%V:S2Í4Û@=š•Ø-— hˆÐãMS J/RLâÑy&hš0e"¬(JD,磤l]¢¶2Yc¦¬4Å: ÁT&u6”ËÕb©CXÐÁ ¥‚¥+¯$%ؤ#gPñ]CL£ì>Ù{€+¾R… @U<3ŽXdÂL +Cz¾ú§ Rãd(oc +ŸÁ";Äìâ(PJ-°†yüЌ%¶<‘ăa¡73 l¼<1ž§ €Üf òdË?'Tš‰C4‰Bd?6‘1ÄVi1q:ílCwĐßp“ eÐʘÿQŽŠ¹=ɔ´–ÚJ)í¶3«Î£ Žçé+Óב¡I»×›:›ã1Xh'Õ+™¥˜"TRô¤2¸Ì8);q¡:ú°š®Ã4t ‘*ú +¿Gqé—U5—ŽšÂ˜V¤ÝFD,P&ß±Û[ lA¸’víÑU/™Ê"Á2uˆÂ5 ÊyÓÛFۿͦ)›Ó¢9³mê$¤9ÅȬzœÞ=[¯è&½ƒt]{ú_§û`¡’LUÓuޕßñórƓ®I1,ð)c'Êã–…^WŠÍ„°WâÍãl&özSsXóš3òł¢O‘=~i¢d˜Ø8jÔ¦·¡ú^ŸŽ"X2âùý6Å`ê‡øý.e +|έè÷è1¼“4üVž´Xl_ ^i´ê,0g“u4—bFÝ7?±œçmΆwŒ{cNï<ñìn,2d½U2K†ñBz»î +8ÅÌ7aÍÿĦ†w¹œ°M2®¡Æx²÷¤"<¼ÇDa;-¾ÏKÃoLqG=‰o@Ȥ¢¶¤` ‘ÿ0•<¯z,X̱ôRó;Ž§ß˜aŒ6º7´-”2½Â[#v¦´Ôé¦wõ9Úóú&9ú9TYº"S)E©ö.«Ôž“-Õà_Çc+b ‡+å€û+á¥ïÕ­‚èÇW@ækÖ~Äðxµw|m¯ð˕ï[Ëv"žF›|ªPƦS&k¼ EOi3ÑÀ6b"Ú¦³±VF‰yè‘ÉŠ%`uêÕ©ENJ…mÓö«­oÿéžøÍ}Ç1½ÆãÁ£¿Çx%²r€Ñˆ%Əž³È/p|gAÁ­“‘W¯uÛõ¾jn¿ÓkÚA½Ñh˜7@ÝÇGšëæ˜W¼{p²ÆÉo8øI·¤EÎÈÕÇ: ‘úáÂâ>àmÂõ²7ÁCdß{¦]!•eB|ƾ:þk¸?káÅö ¶vÈÄ!Ü°MôŸÝÂ1ÃÛ9>2Ì¢ãùÖÆî^]Ø«»²g:FÌÆø‹wŸîo|XšQ¬ö-)ôÕU7灗ðWrÕ¯;–«D£æ9Ý#>¢<·õXâ]o{ÇL3ogF‹9挙q{qfjýÕy‹Ð!Bw)Ô&ÝÎÌ'ô» +ÙM©©ÛÈQaÂkŽT/P »È=>[®©Ù!ôðò´›°0t´7Ø4†diI.˜yñ/pOºFl$Š¤<ʄÝÎâôá¹ á;¼þ¿g"ƒÒrk¥Ǟ6—L…“ª¶ü#%“^‰›\–”b.$»uRüù¶¨êuÒ)n»¶;­b¸gÄübŠ%<Ñæê]ø[.hÖÈmí41Á {‹-;bñ K'eD至wç2„¯n*þá;/„}lé²H ´Íb 7J(ô^è"ÛJbâX®d‡Ò_ÿŽìx ¶lú~á|4Íræ vØ$Þö¸Ìí~5 áÁ9áSt…¹C˜ñ‹WƼäÛz£þ6Ûé:¸BHQ,#ºÜ«&ü¸p^%÷¤³ì0lqC­hOà¡#nü&½ˆÂœ›‘Øãr±áãj‰ŒxeÀ]“_:Ñ «†i8Oaú–&Ÿ×Ò[Oº¤vî¸ñ*‡Õf×Ða6Ô§Ãå4Ó°Áw±¥üß'çœPx¦aÄFgËVsÀ¼ÚG6£è%ÌËúî‚hÄQÅ&¶´ ‡krµ\5ð¬3軦ê†0ŽÃ×пªº&!z×78ÞV±Ï×ÔÎüv%æhìë‘øªQ¦Ý.ñyŠ*©“ø\‹gõär„.V² Û/ȹk:d¢³,ª”LÔQg ŋê(+ Ôuûwƒ"¬D3´ò@Õ2”¶ ‘ä\XyϦ¹ÊËÏpPÉж“s|fØ*ÔÀìÔ5F ¢œˆ7°bþádlKÏù|õdp"æõQØÕ`Qu`)Ñ-èÒÝÀ4]h ,j]!c ‰O™,qODˆg4%io~]*ýïáÄn¬v{Øö¹ÇJ§^ߗX:ý¤ÐG‹cÁƒ9$°£¿äûÐœØr_Dƒ~rqDyÞ’n]/¸õÃÒó}¯Ìr՛S™'Ï)Ìã)£s’í¯ü±É?;û»£÷ÊóºâÕÄƆsòtÁA}~Ú¿ûb<UÔǁë)› ÿɱ«hÑ)&ÖÒ¬.Å|÷ÅZnƒÕý£cÂfiÐ`pÙ%…-`Y0 iºEÒ²ãþ²øÃíONŠªö³¯‚‰»||±HeŒìE.ô¨%˜ŒZïŽ16dÜ]²‚µðµBüòfF—”'EX¸£1ªbgˆŸàz鱂âl?³9 ¨ùW§ª¼F!&J¥ÞÒõ"r´à” ݟyJRŒ$Ž§ÄB/D걅âóQÔ Â\ü»£St°Ò7œ2úÌ=“ð3i쉺Ȇ4!ñ'Ÿwµon‚ ‰‹v\µ@þ!‡ +A†#À™øI,^ض¡|üàrW› +÷–B¸1±"3þm$Éïè8ºÞàȵ`{´ÊÅ +LkÌ°c)”Ž¤‘«Šf§–s%œÌÖP¹r]ÖRàDՋÂôòÔ3Dþ©˜œ9þ½žºsÕ†ɥòn€ÊðÚ5ˆkòÿô®h˜v1‹Üʦâhµ¾»áŘj<ÿµ× ª’P:ž½¡!“Ö+“L[܆޵‚窾49²'=4·¡’<]º¼£¾Ÿ«ò ³ddü.0ƒF¶tñEÓêix€eo‰zcÇUW¬q¢êÏ÷)šØ²À¬Ù(ü²—\3c~ŽÙL>›RÄjÂ|Žÿq­"O®¸#™ÛjYhÙì ‚ܒܾòåèo%,RÏ~0A7åÒ#„+×O6É?4 ¼Ô4èD±g†+(üÓ¬½p7L¾FÔê PøØè–.¥ær‰=¹ EïÿuœøÌ!²(¢ 1…&¡ÜpˆTz-¸ÑçŒï6JP葇í éÓ[JŸ«¸ô ÕZõµNåƗf݄4‹dR€†ÒÒ  'ô¥VZQG±Å'84±ü{‡DŒžÊè‚1ðÁÙAxŽ 28Š ¶x¬|Xá`ùð—FË?•®ŠbxõŠÙj¡ÊGê\24WVØ6+k ÕIkYYJË%QBþRXCà ݙ.Ãt˜‰šÆø¤¤sà¢ú„f,ÄÚ!£{qޒó»¶âI6(îHZx<#u¡Qô@Õ¤oÏ š¬<‡Ó2ǘ^¦lR–:¿Ä,bº5Å5l’ÀV.n!ñÜÙ¦L¬øÍⲐƉiÇÔªœÌgeî<™’HzU®â< ,‰v¶ËcØvt3‘’T ñïd+.›—M² ¦¤‹xY5DkJÜåe‹L@Á,dïæe VjÉ1fMóÄß3>YÁñ)Ö #¦¢Ä2cÁ´^!À8ê—]VėŽÑ€}³µ5Éjå¸äé@w¥óCÒÁp\ÞÀLg×Ò +‚‡Æ–E—yˆf×3T kóq”>—Ge*ìo:º/n/.ïß E]AkÕÓ¯z+Ôq¥;ŧÿœài¡b’¯ŸÜ1ƒbûÊ®P0ű%>ÊQ‘3¶."gZ…²õÂÆý´…'jo³fž1¥ $lyH'`­ýÆÉ¢ÖÜϜ¹>Vm ’ÙIÇcQ¹Ósö,¦bê„-Ë®8ôR«R×Ò%²=ÆJXŠÃ\5œ9–pŽ€ú¶¢§²8ð/ȇÛԑ¸*¡¾WM¬ÍäC‘2¡ qÎõ…N)û½, ”·JÄme¨ðÐÄv¾ÚXâ$ +Á›¶ÄÂ0”‹õ?ž½Š²Yä ¤Ì13g‹õ ÐÒBÁ'Õ`ºë64ˆ®‚™0+M(O”ÿ6jR¦bÆ%Át tSŒ +€CŠ(€e ¡›‡ŽÓÄA&|ŽÙŒ'½.½.€y]÷:U‚%Þèr¢P­+þ”t±¯m6Í%yªl.„ËËq’rÊgiB†­’·L¦!'E¨K¨/Ÿ¹Øå*mÉpÞڛ#†Ÿ* }ÍovHôê #BH؂hó'_F@êLýp„JA­†.ã¢NGŠ¡­ )Ùwç@¹hD×FClM\ tImwecè Wá6¶¹_êèT…sJ®¯APÔـdž’.­£3ª¢ ÿŠTR¥ë­¨…*Qo0x»¯Á€âcãÃÅæÝöñí»­#üeçloób·Ûf§óikb½>ë¿Þ)mõ \æëŸÿ>9Xkµ?Ÿ¾?o|h ï·ÎÛÌ՚ý“úùۍZ£±V­råA™ìT×ÛGßʵwlÞæÿÔ+¯6ÚGÃ×,9ÝÉߟ†íãõŒwï5¥\ޚ<»j¨ÿ ÷5•þF«ùß©£<|í6þÛÝéÌéw#gºýþ`Íí+õÛuûjCWÏÔTcëÍÃqóæÓÝ}çºW¾É¾4Žk}mÏú_Û-^žo+;n¿tª_mxÁêÿø~á*㯷ͮÙ6¿´Æݩӛ6ÿ«%ÂñcSÑjßíã[·¾p™÷¾M¾Qømó§2Я»»oÖ;£Ý5Ë÷á ÖÝW‡¥míDÛÿTêMߏŽ;{›Ûݏ;?¶Û½­Ïýqÿzs¶6=Ò4<¿Û'ãáÔ¿¹V­à&3Ö´Œogz×Üø°µË¶ïÝÎp´ùSøÿ¦}t6Ý{µqptóµÝ±´­ùö»ó£Js~ÿÎh6+|¼×aÚ ¶=kÕB‹šrÆo lÍ-Ҽݫê-£WÁßÚù»ÒîéšÍ˹ÿ‚‡íÞàxíödçpŸC^wkï›=úmûøF¿kÕÕµ¯žÙÿq_]ë©#Iø ü˜Œ@e0"ç`&Hädxÿ-u $°=sfg¯ö†¯QWWø+gvn0(Cœ†Kt‡®í œ2ÂÚO‡ÌÐlϊ$õá,I‘OÖ£8ˆþɐBCÄ‘<¹ÉÉó[ ‰ŒÂš'¶#«˜\ ÊcÌ,ڋ¾AèvI"“‘ƒQ)«¿š|:¯lz¶*‘'o +¿Š7¥‘P¸)ða)@ù^ëÑ<†¾Å‚4BPKó¯lŒîÓ«)ߔV„¤EÔ£üùéu ô¤Ucªîn‹¯ˆBUÒË#wüHéOn!ÖkŽ0˜}z֗GÑb]¾¢H«ÎH\®èHZÛºuEÔ˜'õÙޒ—l·*_)TEòÏTŒÍhûàýó!\#²íÏ#6(C'?¹tóüÌ7 çËwӐµápuD÷串j@æTÄó“[îÏÜztÌf%Ryç^P°cV‘ :ABØ3ÃG_Ý#kwìÕ8r²‹¯ B l±ãT(æøQBÇôXñij˜'Ùf?êw Y¬È=ô¥ÎÌŸçàt}-Å°bE*$@ooT˜†<3"”ƒ¤ÚzÀ4§_ÔòW"=mÕùD?ÚyôA5¿ißñ~ÎÉ¡à„ùÉ%Ìz®ŠOn¾YšPa2Œ${럴E”6º\/¡AÒÈQ2š‹•¾Gιªø«›´"ÅûQ"“‡žÜ–]`ÕTSd)žèP¥ +N˜Ê·ƒH¨Âê<Û +›]{Ï7]°P‰ƒƒܗ£Š'”ðumá…g&Š \-ý’æÍÓ í~« Üîæ¯%¨Í…Ó_»FK“آ÷ÜÓ¯{:_o6ÉÕ&Ä¢Šf5bÝï $+¬}‡¥Âæ(Ÿ­¶÷ëO;&ÐKlŚ"'Ù÷zìXæá"ÙwNè Þ¼ñí>9&Òo_Tì©3.P—Í€oŒ ¢yËðcŽÍ‹a ·¡ +=Õ¿ŒøÆ¥³n1Ô1ø°åNӔ{˞Ÿ‘›­ˆ»DBýt’…ó€#UßV†Ÿ™@ªÔŒÇ§BS7ß>9û7JÐ b§ü4ª>]¼>\ÁÉÏTñóõU惙 ËÖᴒ0‰I|ÀRnêDv>HÊVzdaSþö¢°žI¦6X%C9Ät0¤`3GC)¿ÐÀßLfÈ*$ÅxýÑMåG¡EÖÄø![YÛë¹ÂQ+ ?Þ%ïrWٚ Œgß[¾rò² åÞQ?»öŸ:âÁ 7)QGî +(j’2WœîàÀtÖ_/(r ÉßuøŖ?q ²Ÿ®ÍÛ»¿FX~ù%0þ",úŽ¢Ý*+T¾ƒ…. )?…mEH\á°Ùlÿ*Š ›Í&€7Ä­†7¦¼‹{mlIŠÍdî„ü°Ðnå‹å­{¯bì Ž˜ø)Ū4ž `R$j…ýbÜÿS¨ÿ6^ä]lšc7Ç[opêâ†È%¦¹1”Ɇl?]¹žA†ØRïûýèà›mu§pÞy&}-jo ¿ƒ^©•q©—R0t1r>I%¯Ý>ä‚ÉEZsgK—Ã'O×Gne¶q"~®þ¾ÎØÇ¢ƒÖØÅ¤‡üY<úUßüïV»Û¶H'×öúqq©÷%U= +vƒ„Ñœ¯ï¤ +מ&0Án¸-u/…@»CªLnFlh¬ìo#Ïþ,¡íƒ< 6#Œ†l{#%LtióÖleLJ… ccNaïɊmu³°š»-³QöϾ1èÆàYh)ÕßyC/˜”`Z÷ÿÍrò«‰1)ýËåäVsê»EhÈ0m‚ç_W‚ÑíÚàoŠÃµµ¡$7áSÞôA˽6ç£nA@c¬…“Åêÿk³Hœ–®ñuŸÊ¿Kª@{A€"Ô¸A@¨¹?Îh& 9*ñNö]…}†VÃÒÛ*ñiÁ›…œ ?¿Ë'ڛb“¯´TwÈNœ–"¿-Ï Mn 6´>@Óÿ“Ûšÿ¿+:¯H¹µ{éUƒ›ÛLá<çφ!Ò±ò=Ûâ/K +ëÅ&Ç76¯õ#5­g+Q®å`ÌMµ˜Xë˜Q¸¿çU´¿vŠ‹wG +®Þÿƒ¹aÈ}a]"`©j±[‚¤ŒÅ`(¬£¯ëâìdj¶Ý†› +¶ý¬R¤…•ßÃ7. eøp*òîYu3Û$Ž¬ +2H&¦Å‚P«rÕ¨¿·-›Š^zn¾Vn´É?»^tV/&Æ{™X_&U²xor_•ùéA²™•²ÿגŸÜ·T¸Óiъ¥J,&³é±±Ý‰¤Æˆo6ޕƒb¤‚­/ǝCžI«¡ÛE3v /y´‹ÉAîô\È+0åªûMp¶ƒîTW”‘ß±0,}áØÉpÇuj„'µ Kh~Ì +Ź­:WŠÕ0Mřš¡ó!©p +ŒqŒ}‡œÁ”·ü½cögÙ¥yê€c›¿/’F&½6EŒì…Up¯‰Ë¯% ¹¯…0?ÈGñ#³ž«¢-Ñr½„6ËQ2š‹•ÙÖй¯ÙûBåDuƒHRTÒQ†Ü÷÷ªg^íQ3ìÆWµ<“f[&x >ß³¤y³¤$杸…öŽL8LåÛAÔ¢ åâ©ùØ37qÊW#Â&\§€ic‡ÆŽ'7–Xn%?äÑÐ=åuw†¾[ÇZÜâª.ír̛Ñstyé/ïgT±t¿ø4ûuQ‡I)Ñ)÷½S&Øy.+“Ì&i£àh~yJï|ÌÇÔ üu’x0Å¿896ÛòHoÛå@ˆ·»a¾¨ê|½I(/îÜGBXS³“}TÁ å5"½ Šcب#¨Îlã'å“M«¡Œœžž¤^Â&é +JŸß‡†›[î£IBw¯!&˜ç´æj,~ƒ&Zÿ’‰­Ùۛá«ÃHrm×ù|“\ëú\T`ãâÍpèîÊë ß9Ödq믆ʩÒJÓRKo­#]W¤^>ð“iwtqâɝþØÞÿŠ¨H”§éÊIz>ô¯E}!¥Êzz)÷} +*cœâqå#·=¼ÝêZa¾E‘ó zIi™,Ý͉e±}ˆ ڙâZ¹Ì(¯®%e¹%ú¾Ã˜*˜PÓ8*ÊèŒÌ¢‚V:çÙOƒ çsþpc!{Ó¹W¯Ö_@›-H2}´±Ã«(, +á•ov¡9+/r…þ¬âxN8Š·­¥$øšÄ¡ÛÉÍÛã}qåò¤ °“ F˗÷¯jÈ̋Ytýì@ÔçhŒidù*M²;ó¾ZFh=BœML¦™ô@yçæÄåûêÉ÷#žÄZ÷K0÷‡=f¿l„ bÏÅr§KÖßï‡ñ`T¡l«’|O¼*bÃð“&Zmqó¦ùWkE·Pd ¡H/Z¢'+3ºa^_aBkº+•'ŝlÖ»]tVVt栏ŠËäˆüS †¬\A},ÿ©§š÷,…*më;$Ÿ.‰0_Ž}²wOT€’7˜F‘mBu¬}tw.m²óxåÌE[J„ξ¯76r?î´"SkûŽ¶v=2ñdÓï)üþ-ºT‡ã[¨Õ¤=bã̉CÃàÂS§‚oj±wdw|êÏziº`|mú&S"cÀ¹¤¨š„rsI‡e›ïlX6ã’WpViÝ=˜=3¼Ö‘¯f~ö„-²u$“Tò9ÿQ§>•ý%/¿~ÈÍURB ±4±¤1X¼ 4¡¨Æ?ëé)ŽÇÛåðÞ$tø¥J=zËh3ÿîÊ.\¥_±E¦Î€Ý¦iµJ@œšà Þè‚C¡³‰­ldNéwO&E‚…ËŽXƒ»ÝÆ&ÉÛ½ª1Û؞òSŒòÛüU"~9`¶•i͗nØÝÓÚ×N\¸qTnLis¾[mŸ/–?sáØdœºÀgZâO²q[5Øßür¾óùhý¼Æ\è¨bóQ+Ù å/8)óHÏ@‡‘¿º—¯CV= Yˆ¾ÒìðÞuË4µ‚vþSXû÷¾3[øX|牱5P¼Ñ'àhiº” 4ƒ=sCyGWî,u%´·KÂh‹â€‘ø–N.x&S2Nl$³ŽJ¤[Œuv©f};ð¹´kÎw‹41¹0SŸÑòÌd¦‡…m$X€ÛoNDàz{ŒœŸUý'ôîCjûâBmoEãðâÇѾÝzEÞ>KÊß×/n-½VÌ= å#ǁPu¦¦!/‹f×b'?;»FL&u™„ÿQ^0Ž4•MnïvÝLºP‹Ë/ÅenX.šj&†\µJÊ÷¤Ñ«Ï€ O“ƒÐÎtÁâáã P¾¿™;?0­mJ~áÉ'm?]\µ’€²w³‡W€zA<þÑOK£³æðÂÍbŽ[¤p>¯ðü1ÄÃ: +”0»VãŸîÐtä+Ž×¥âÇä贋ƒ%-e€!ƒbº­ì)`ãÏ0r¯o>«(3)Ÿ„>Øm$&•>ûŸäåÄÞ£x½ò>œž…ñà¼äÊJÙoíF-,%­Dؓ&à‹•9‘!ñ£MÆ۞$ã;kfWÊÂn½ÿ³Ø.×áØó¬5ϏÓKÉŒÔIKØz]Ïw²”^¢<–aD0ƒÁ¨÷ã5/Ábý”Ä…VèŒ +÷ª,“ —My¾øbõêýù[r® 2ÅΘFUœÀ\ôÌ8ğLjº`8¨_äDÞ¯a{EùµWà‰ýŠ~.;8‘Þ‹ëc¼ÓøZ_¦&|ÿ*[žÈ=?Ã)¶ ®ÒZ¹ ››ìGìœnÂܯ ~(gOŸô2 6)'"¶ì¯†á#qF?V'…Ü7§`¹¢bÏúîRB9è—iv“ã«žÛVhw9éî¶&ÁÌÙý&"Ñ,.i%•·EùÇÈÇâÍo·¨²ÈËG?¼·ùoøÛØ2‘aeîŸ)~*ñ©2>iî›ÄATz"{lê*'Íw-7ŸÃ›–˜bÇ{9—å'ãu˜Ñ +ÎaÄÍføæ*Î9ö0ï&{4$Ë ±…g?;SðE®JíŠlÙšß›¯xÖ=ÀÉ’«“hz÷[¥Ó¨1O0ú¬C6٘7–"F_Gè:$ =$ü,/‚¨Žïà!:ü—‡XȺà –‰Ia‡Œ´œLÿ—wxGä[€)Çâo•ýô­q8­öšñÆMN“à[NÓétê -k§·Ú«ÿª—[äú¾?µÔÙ^WTåò +rT&F¼¦pûÙy§j'S¨ù2±êb¥IêN·3j¾·¿y˾¡L»X.gRÈUTS*’OÓH¡;E1©#HayJ˜Oñ$ÕÌ&îÍû“u@& +҉ÞPڈîVnbþüý4ŽäZøÒÔuÌÌwx­9u"³¨Çüyw†—ë^4Q#œÐäoÒHxš–æd¡1Îcâ¸R0íÊ\¸ ª³g©×d®Zv¶o§Ü&Š)‘K¥/³j &1j,¨ìà£Np}®°$ ±L” î|íâ ¼åø¯12‰¥ÐRÎb°yUúo½_üéhz š¾xá‘òå Dž¶3 aóïK×ò‡ u»+»«““ûäîÅ¡#<µï—FjÑÈš=¦øb–‡eú·±È€æÌɗœ¾¤7ê7±™RB&Ýsø[Ú·=Îr¸>ŠÔ ]'¾> ?šl쉰”–bßÉÏe-üRé`ÁŠ¿*ÅãkÔi՘]í]éT¢ˆüZé;twÛÚ+¥†{ˆ³?”ZZ.jÅ|óÿ¬WéZò<½‚^KöB«¼ €PÊ*È&È&*n¨ì÷ÿ%-%]’RÄ?}”´9™33gNâ?]<¨è뮞ë5,¨»0‰ |![¹¡hB¬—ïÏڗ"€öÇ|áõ¶…•]×L´Ö»œ/½ò~Wq «¯dšzéñ¬vq +j̐Õ_¨\7° îÂw‚oów8P¾°ø(A)š©Õg2>V±?ãå;Ï>ÒÛ{žžGº-_š›†•'´â@^Œ°«y«©‚JÃyÁúæ«¥„Aý–H‹óU\ÌfD(P˜ê÷ë Áš@ÿÕ¤ñ tÌ׿ê-\ äÕќ«#ƬëcNìÙ¬¾gÄU0ÜÓi´ja,*vCSòשðk‡¼šMˆ¢ cÑÛ´° ]_%¯ˆ«If¹˜W]L w›¬ÍëÞZz®­>mÌ«1>ÿM\ \Ǫy´ja,p}Õ,¿^*‚´¿ñ‰AÅ"6×>Æp5«ýà®fèfqQLµóóh.)ßç=³r+Ÿ ´ÚÿB.fþ’7‘-›+ô†…0 Üyå3°…‡=à¡ §'ò‘zòpK`dš¦¹¿vÇØT#´¿îtû ]¤iÀ*Iõ²“Ì úH¬#µ?,IõvÊeñz@)Á‚yŒAùï¯@Áe'Þô ý¡’}=¬.Vwa)AeÏC‡ʖžÂo(tzØk—O*¶¼z‚1è Ãxtô²q©`½ÖÇe_ÖHðˆ ê.LÃDPåæ€@ab…w‡Ghb*{/‰ `"û!V`,€#™’@›6ôÞVº†¹oŠ:cV½p5xø+´7[þTÄÑ{BŽ7ô>éMþџüÏÁ{ŠQCjbZZ[—UØI'ŠÞO¥úÓ ¹—‹±¾‰TfÑgé Ô=r!ýÙk’à½(à×&ê'@ë…/]lnÕsƒ¿€šÈ.IAFÒ¤Áwà_ú­_P ´ + œäE=OƒYj/µn”-ìñå šFô’{iïý¶²j\ð28æÍJ}E 0v8r6ò>ËÓðá9°ã×˱.ˆ`N½lñ@`LO`Z¢÷`m¡àÁ~!ª°1Êx$O4؉ác ð xèççª#Êaö9©rŒtå1íåáá}ÄøïU¥Š‹¢õÂ)ƒZþX·ƒüÁþ¼G÷¡ý|1E¨X_[²珟}κʬtL–Ýf›ù=q+ŠvPî‡b_D¥ÑFvZY¨®ŒY­¬«ðéÌxŸ¬µJ>›ù×\x`Ï;eC–Iz¤aÚo”É,=‡èMÒC-Ý~pm ðpd£ôÌaF5eVî$XcóDÉæ<“ ´½û•†ùÐIaFR.ƒz…9´¡¡ÎøhÓÄ¿FÝý½ë;‡»)åöE“ªEP%'WäͲ‰²ƒÍ°[AÞA,¨ï¦n9`컂Ӿ³ï:XÉO±}.#ždPyìUPv1ز˜º'n\:Š!¡´šPEýÿ=I˜Ú( ÄÔÆ ?Ûv"¦s)±>ÂÉj1b‘÷E eua–”( ÙßøT§hH‰|̅9–ž™¢£þìg?˜ŽzFýËó[âX¢è“ò ?ޝ½/›uÚJ–_?xÉGš¹É³òæp–N¶3tõ¦’:ŠvšA¢¥Ó<…½ç×g0AÛ‹ò×ÕÉö¾’ÏØìˆáQv¡h‡›EÎ RS~m3þï‹þ%c±¿dL8±½ÓØZÐzq}+ò/ìT:Ý£³Bo O›í½Ø+‡ÎÐâul›qÓ1ÁÆwå[Ñ8ãÏéÊmÆëÀî+=; –Åf2ÛÛ҈˯Øa_>·4³ÃϦ“αKÚÞS`!ˆe9Ö/ÒdÅ3:%üQ¢'q‚=ˆÅ[þŠ“#Š¢0\ÑÔ½•1#ªšøi㴑`ô¡‰Á¦²ˆE­~.¿->ÈéN;Ú¼mOð¸û,#6²sù 5Èh³rDÑ'ê—p¨±ÃëÀ+ˎ`´å²‡9t!Óm×ì)§’þ~d"ëÚãäµ›'wSTü+»ê!e€óŦ!ááÅV2Aío£S/[tPŸÔÑ nÞÅsÈÒݒJ'·¸×âû€¢å©Á)Zð&–;ÒÍ +à'dã»llpÁx¯Dʆ(‚7ÉymOƒ½T覒DzÙÁ²¡¯ å1íåí +í»Œü»Zɾ:Ž„&rljfࣩ»%™bËø¨|Œ¾Ø£år¡„îš-ç%VÆQvІ:(³F˜ÙLË+”áÊ)1DZÊk’{8FŒN +’+";±ìC¢|¼’‘n> >D‘ÑJG¬VzY1Zi3ù”3­æ Ó x~ÉÝޏ)ç1Bk1Ø9a=wÉ;ÖTú»ØIƒ„–<¿_–ã üMg:´êAÙVºmW,sêaH•jÇ8ô~#É;ðc†y÷Kb.퀲Œ:‚—­Xݱ9¤ó²ÐG]Zn|ךú¨ào¶}gßu†J"p }¶‘iÍʧsJGæu³¤i­_HÆCÛ,µ=S“{-ؕɻÚYµªdÍväQ´ó}ŒÉSNgåaŸ³;PÝE»M¢]&Ÿ²Ï©sOuJ„ÍráÁo‡¨*Z À¹ÿ9ãLŽþ¶ïœÈ‡'Tû6”] NàV™ˆBcøêûˆEº[pžŸ¯û9{áÐ9ÊÒF¥J*í'ôÃ\RÚ{óhɼ/£¥ª=­0Ì ¶3O3¼•[¯Y`¬°Õ’…“ÌZ³À˜ˆh X3Úx,G”°e`ž<‘ lGˆÚ3­€\l;£åwd5&i¥iôeK;…6%wÙiŒ Ã~ᅤãÄ y Ç«0üRUʅW|u°H†ü&Û{LäN¥‰$Êïi·QÐØL;h´[¸ý]ãò{Ñ+íìÎï+î- sÅ ?¿§Y‰(AvÄ®!ÿZæ!ô5Äoá>q -/ÃÐ×ÐÉ~¢Ón¡{+3µd!Qéfì'ˆ €¸fÅ F\h)M8š“Læ8ñ¥Óµv24ö.H¦ L-b„)û»¯˜Ýu:inK¦"šèB8·1xM9™Ûr¯öѤoyt"r" „Ä©9ÉÁð"ÓÌðŸ%ðë,áºRà0Õ³*“=åÞ¿Ò`EËX›½{‘Éüg†:%.‚K)®ÌeqrÉéïd“À%™ÆÒÕ¤ÖaöÅ[ŸqkyBc)Ž1µsµÅ:âV¬XüìŠSŒY1Åêd.âhŒA1 +µá„Êô½‰€}±fB((Á‰ Áå\d º6,`#®ŽèʹbٜqpqôöôvÔxlîÏsëڕ¦ÿªL*? µ^˵ò¼}ߨ¦ï²8ޏÆ7às«~™¨7‡ãæk!q&5LcÀ¬œ@Bv>Éðž¥Àʔ4ãøºz‡2Ñ%Po?]¹;|7äÁ¨gußÁ´\LÛí¿ Ð㋳rýiäW®{ü™Èç*Í®—RÃlñêí”æTß¼”UÂ)T¯ŽÛ|3¿×e8=”cGw©g†ÓјéԐšË"®^±\ó-ù±Ïtš¸œe;N[P§§Ñê´t* )p8}Çe¤À‰£¸î:-bNÞSÉêÓºÓRê9¢n±\§Y¦Ó|³sÚc8=”•ÒuãÅs +rÁ +<á8mÅÙN[·ò#ÃéñEDQnþëN¹ŽîØN¯;¶SMë71vÜÆàß3î'ë—úf¯’ûÏø]±žçÿx1}¿ì•XL—¿ÖšG&AΣ’õ~à&<‚é)‹aݽ ”%åœ?ôªŸÈ>·V”ƒ&PԃÜ¥×k %ÝÖ)Û¸×ÀveIcۖ:¯ÓƋ&eÕ7 —ÖV!¹*‰ ª®³$@Í&†È‚g‘ó^ÜðXŸ}P‘[Ò>P[J€ÇŠv¬®6ܐ”ˆ(ÿm×~›Ö9ŠÏê^ú–:}ágvŸ<ô Dr±G uÿɯ“W%G¿pŒ ¾Bç½€ q™VØ=¿ñ*/ÂEÇw„HHö­ô53v­±k´ñٞ•ËµkB#wŸ6 ím|‰À ’¡!ã†èdù—ÆY­VÙÐÕjb[®ÈÏGûQ&35׃÷Ø©õ Zj@ñiã‘DhÚíSãSwW‹µ:q¦4Ò"YÍ{Ù8µ[Ý×|àÛ)¡ü[b5PI©¸¹mÊ,9œß“!þ²`Û&@ˆˆÊh +Q¹öÛʘ]y;=D8µ›6󯉩æå¼ëNnn%z-T²dï´YxÞÔ:A€]Òk'Ï ++IB”\Æý]šT†OD’»¹U3}5ã>Ö/Õ©´n¹—˜ö+­ ¿;4¤Å±ŽM&@Ŷ¦ «!scŸÓb5÷بAéo+˜v%ð*²ºûÚ{ å&ñx+Z¬“(­Xö¶´‹U™%‚*ÔOÚôxTÚå–hP23(¯ubý[Ä̟ ¨$o.ÑAvÿªåÛ.¿E´ƒùùtÒßq`ÅcóÁ÷LV~n,ìIoŒ¸*á*ö¦J£o«Ø[9ùøMs7Ún+eîP©»>[ƒòTG†¸6üeà|lG±>Û0qE@=¢0LTÙû‚0Á2 ™b¢7TÍ'ݵá*& b"bïB\ìù©˜in˜ `dÚûË^rLZ j2ɅìЙîË^í£)¢yè°ø}Œ}¨_‚ÊAq”±gò1Árð˜ Lg.xB]òŠÃ#÷ý<òãç‘Ô’G„G2&ù2˜g`aú¥’Í” +ÏÆJÚB®Ò)dºšÔ¨×|'>ö¼v$ñÞá$³ê$Œxž¤!¥ÁûŽ:5Ö¦¸bŽ@Ä'%Ø#@ˆÂ-BÂô J՞–È* + ³ÀãÔ $ìlÅÇ jÆä÷A!Y[ìûŠc'‹üÇk×DN²ÆÕt،ö“´+Æʒ„Ϙ›:6øqkðé+ìo›£ãÑQDd@Nô‹!ÑÞCƒË”¬äÔDèԊÕÁÏ.‘š5c˜âmYqS¯èß*/’Õ¼—{GËbJ7XÏqðí”q?M]‰ÏÎKlЗTÜ\Œ6µ7¸Pô÷+‹«Å|…9( ²¦ÒºE\ÛœÈ$Ëô‹Œ“åj¹·&Èr±ÚW¢¸âó9"ˀËÉ9GÈqÀî@1{É]"BT줵ìž7µNP/á°AË·IZjèE¦M:™Awç¿/åÛ(~ô&”:Ÿ çE"!tÃ'tÖ-·®«’è.;¤Rݸ ,Þ²¹° Ãà›xuŽ„&™…ß%jñ%•ô“×á½è¡s°Ï:uùùt’Þ~ 㝬ÃI~)­¸½”ÕL_ÍÀÇbõÀߨA3L• š)0É,ÚÉbÐXezÆ@½w:C@kÂj÷1¼ìjÇX<DcnžB4î ðV4åä#ӊ9cq[öÙ[^qŒáò‘’ß”s‘]²iþO~þ1O—~—å#‡)³àN&`ø1âB,†‡›Ú3—äc·‹YÊûO~)ݾ˜}Z8$²5ø“;Ì øî51՘ûµ˜6^4iˆ«àœ¿‰ØtÐVñýïa–©jRëpK¤€)Ž1µsûƒ/ƒßŠÀ%”Ž(¡é„iGÝ«}4¹Ý´£=­£qø˜igό€\%ejI{€v̄!˗vÕiüÍ­{]3ö¹0-ý0ßqaH€0¢°a¨/õªl80Ah¢2<  Eö†QòØ8¬wx‚ž!@ˆ{Ù†BêÕ²†ˆÂà+h‡!ÕJD„Ú[Æj'mÓ¢*²q“BÛÿ“Ä¿•†ˆ¸F3•®¨4¶2Ž–ÄÊp7Lk¼ú†F¨’¤^™D‡bŒTŽ|šûÂ":ÀX)´t՞Ç•´û¦z0I'ßJDÐÎà+L4 ÛÎhùY©³Hj±`;æ58ÏÏ×ýœ=C.¿›Jë–Í“ý-ë°Æ7^ãsï_é +@ –aaDxêe‚ïúüC„Ý!k¡4ÀÛTiV)‡0— Æá¹÷Å#ðyòŠI+%p:‹ò“$L£½/ÀØ»-üx!Ü:Ne¹¿üóDwZ=2RóC%œZÙr›o¾]ß3ÆÕöä_–ÓWÔié>†8=¾®ÝÛWÌY™qÙsjÈïso)1§Çg«Ø_7ÓÖÌijD8(X+¦Ó|ó$Ípú?ëU»›6 EŸ A $@`±¶¨´4 tŒµ›ZJʇ jªªˆ­€xÿىãÄv®cÖþ‰°ì{Îý:çä8ÿ{ḿG½™V€±¹»&ühÏ\ŒÀGuß8}EX’¥”«Éý¡ƒnw÷íø(ꗋYõŽÉªßŠP¿H"JWÏk•sóýúw.èîäÉñfîŒo2oüöJꎬÎ.Â<½ä„(uI c-,Tw¿…‡uS ¥b¬A°´[ýü[4W'v¼±ÓªÒ¾¶±ÿruÑ_|ï’'ø,Ç6~¯ Œú‰ÍÚÑԐ´‚RP¬æCÂ,± ½ªÁíBâ^QÞÜ]rõT†–‚þM ©c8.¤ /0O%'vÛg¤i~'NúÌSÄÒyY+€–‹îxƒª'¸êþ›?C¢j\­žêþnõp8>fòG«Ÿ„OPãÿOú…ð®Têª6ùœ~9o¨ò”в¡˜ŠBBåÚ.ãúüI+ÕÁÎ!s,ÝY)ÒæL¯·rߨαÚÐtÆæ·}b,<îUs +9øh·º@óŽ`‚S€öµ3ºÊ­ÊÆ K`±_œ©]QAåß֊dY ÓÉá„?fL̖ ú‹¾K*'(¥b0PB+z,Ï|o,ózÿ bRh©´ ,ÙöCîÔðe ¹“Ñ +òËX™PEÂÒeEBJ«md¤V­ki÷êHÿ MÚW—ÎxW.]k¶ï²?„æßôœ>Y®q%Çÿ›]ƒ•Œ¬"‹£I7¿¨R0ÑF¦‚uªïÉ$/šA®Î¶Yó5†Ô½2Ò0>ŽPAqsG+7ˆ1t¼‡çXOºA2ò~î±mf±ÓR®ÿjuI endstream endobj 23 0 obj <>stream +H‰¬WéVâL}‚<KK0  3â[tDá ãȧ,÷ÿº;!éÝIGøÓÇ£±–[u«n•ÕYLû< •÷9#½O»SEUR·÷©Âü<”*§“eðÉÃUªpZ™çŠwϵ>i—ÔÆ©(*ÿ.§ñŒTT¯U%÷ÄTñm“µŒeŒ[ô?…s}¦gJÎð£lîÇ}>ƒ{‘«2öXþØ!E€VPÊ¿H^vžÂ2>û*衘eç|œ‘.çQ󓝃Œ\ia”Qy{©D4¹£ÉðWúÀi¿b'yg®§Šùû[ô‰åà¹QÜeßi©õñVþne,dEi`©/“q'¸1º©÷—.@ìHƌù#Ӕ 3ö*­zGCìU ÷”äk>þçhˆ½–O –“ÛX£6…•:“e¬‰K½'ƒÙĨ…:Pv¤ù¶¤3kSØò’f"¶)rÏ F Å°·ÀÊ­ž†v—z+6»!0¢ý´Q…´äÏië1‡ÝõTQkÞxvDL¯ÛEa™ø±¤Œ‡:m<  `¢ ·>¼+m×¹í "ˆ&cåÀ + -Yz¯¹F9èdóˆÒ€v´ø[÷ç>  ÇpŒ, ÈJú‚Ky$äÅsC.Ü1/.äa¹µ•ŠñªÙIðLuñ~m´¯.̆äS $ÑC]  œÇ§M<æ§.8q’áä {ŒÀÉ_Xp¶À›æ„dÍ1Þ Ü!qŽÁoãJç@!/fP@'*™ÙÆ §›Ü0ÐÈ4·Øêùr½´ñöÒäÞ·Ð#Lº"t¶ØGC«Ž ݦav†èћ1æLÎFóqbiï ¼ØÅS’7 sšXBv ,Ğc¢õLzï óÑÈF/š7¦–‹£Ô "9R‘ÄLE +ÄMazV‰˜`æE) æäÿRB竬kZîõŽù¼´Nb,]ÓØ[äû°\À¸8K²Ä +8¿ra³Ó0ò\e Cìs"+Së?Rá¼}w&̲ dü¤êáÀ–PH™V‡ó;ÐÝý´NËyÁÛÁ£ò÷>ETþEò2z\’Ûâ ÖÅÎ7É»–LÑíN͖Ü;ËêtÏÔ¼ý ÷ ½C›{Êû Ä1æÄ]ù¥„BCVð£n±#žJÜ&ò2^ûñЗ€ft2‹¿¤þ…3R ”±¬ÀïØÍ ˆ«nmïÉ;ïJړU 4©[)S€O·"S£m©‰L™jœ1³àã­7¡1Žmç×coíÐ+ ÇøÁv •¤_õ9í¤¾ v%!;òQ²J»¬ì¦%iÛ%“ðFg;á ™²æiŒ”š‰„kóAæ€å1?¿ö`Îx+Á£HF̀9÷gÑðwp…ÇÔü02®a†ï.&”³XF᪠~Hzˆ ¬ÉÓ´ýËñ~—_¸Æz¬´a~[2ß]’îð®¤1ޘ»³æUSŸ¶ÄdœaÐXæÜx?„¿(œ+©×9†œ@Vº~Õç´Ó_ŠÆ¼+)vüd<_4 ±]™±í\©/›Žbu‘ïD»P‰ÂWæNᕷEêR°—,–—€/$Ý4¬ËÞF‘†ýe ºš]| Ý_¥•z$SnæøÏ1 ±ìájâ=ŒÚh]DBA<†=¿^‘€4¤n±îQhØÝۅ;¾”öÈØÔ cš²îJËXNnc-óƒµ æ€ßͤIÕKJß.°‘QÎk˜¼½&á¶q(€ŽÌa]¯„Ðñ”92Gà:–1­yóÅ#ê±ÞÙb÷÷í'•5´pÚ³b{ `GK–Þk¢±ªï¶sÀüŬL¢RÝÿóµCž‡܊\Š[í‹£”ò+Æjy-Ÿ ð©ûÜ(î +w¯ÉMFÑ€„æÞ·i8^ïÓü.0 ]$4'Lä5Ø°hHÈg‹›‡2w_¢øCB`'ª²/”ã5YJ’–)§.Ú'ÛØh[júµ|,S¹“•ãu6zѼñ3ÆÐ:$bgǺ+A%»î#Õë®d ®žç¦ & ‘ëº{‚Ñ\Â0SêõÓ.a¼”zOÞ»„ó>›ÌÁ„,6 üÅ¡ìÃ\ꌀø ¬ê³ë[êͽϢˆËðׯ0êRê­¢þÛW4·/*ròç´ÊˆkCÕw_Ð0ýT1ë·5ÇÛ¤3<ì+i¿–êËdÜá4ÆXå­<È}ىõà;Ñ@H`Ýô™ªï¿àOŽ`/Ù~cPüYÒï²}öJ6/´\i?†äîӃ †äñìw(ùûçü©*ªJ7ǧªÒ›³ÏåbŽ‡W¾£Ÿ”Ûª\ö¶ëõPü${óúÃvC!Z%\Dã{KQqDÇë0À©•FARt±Ä]$+Å8}‹O£í¹Ó¸Ó\bè85g2æÖמÒL§j­tvÇpz‰~f/Æ4§‚ÜöŸ˜¹a=tÊvªëë¾í4K8ý^¬Î¶CäNÐ$À#Üiî!Š9ý~ÕƜÎb±ˆã4uñ_çÍ© "€“WÉå.WÒi®ßg; ×þªL§jmT7oˆ#çJwÊr:ñpª«yšS€t»6þ6" €Ï"¡aä¬Cwú#${ezCNí ª +©™F Ÿ¬BüÚÎ\Õ§wya~grßú2|wⰸގç"¶&aÎ{¢sG\ð¿rܜÉØòt(Ê@jS¶]œ2ŽvljÅ>€.”q Í/<¯R¾Ý܊- Q™Ó•z,%ÑOÇic„tµâ L¯ Ö® \!Á~ŠÛØ%{oPJ‰yƒZ×Oé¶b[fH‚HÇ©%I œZ¡à¥uÙ J +'g¥ÛÕÇÎ+¸Ôá>ÇÄ$».RÞ!¡Gàì§ ùüÐäßeø*­zŒó1X~'~|áœ)xC‚¹x¥‹/êëMnÈꮃâ1/ö¸ê˟h\aV‡å¥Óñ˜ê :®úòšãS_Z'1A´Žœ=m]õšÎ&ò’ÇtÆ{z,3ږš~7+|¬¸s Fjé035bÂÀäØ©ñ(ýçD ˜íU\˜ª¡¼D÷ËJoƘ"Ô¼’¦%¼PªåQ+½¿ô?©bæIe— NK¢P^'‚0,ä:æËW 4VúsÞà8€‘jÌD~—$8põ©Bj»PÕ³Ï9'Ì$ÜÛzÖ b_Xä¸êW(å¬AÄâÕ¦o!àcu`‚A¾IÍ=A±™Œ?çìs"¹ltZdèd%y¥¾l:Šò/’—Ñc#6Í&Ĭ&¶Ê˜Bþ¥Bšé`2Nªôˆ ÒhþÝ`}'P'ZØ 1ZuA 6‚è­H8ë«CDêÌ¥¹Ï=\gº¯xÛ-#2$1"¨M±ÁZj¾jœ ɞöæ-FeGQÎÏ£~K”Nù*øyPýv¹˜`魏c5ƒþðIîýƒŒu}.dʼøü‘[å@Ć>‡O€$Çë#"FN´{Ù°Œ]¹'-Eî+l ++—>̔zO>r©ÃMaëìJ?%Ì0Q q°îJ– ß«¬ÕÓÐãMkà¥Ô[E=í§m®·äÏi•R¿zª¨5oh3íJR Ð¥XÝ­-Ù˾Pd¸ؕ¦ÊJ€ÈXaÂkédß" -Yz¯ñ¬Ê"@ˆý­óŸ”,<=ç† r‘ rû¥\ A c† *Ÿ xR—ô!C:u…¸iÏ~V¤zJ﫧÷koõ$¸%‹ ×Péêü™¶$É Ï%32V჋qûEñ@ÌQjÖÈË.F<§ˆ‰õ …tnIÁXö#®&Ò¶«™¬A ®&g:÷Í=W#KY=µÊ츒mGI~!7·šµø Së¬ÙЋ#ö3[..dïáöÊab³AÂ:ê\ùmbí[œøéÊQÙµkÂ]öƄىwSvˆŽ³N•ìÛ­ ¶’]©ÒÚØ»œi¨½TR0Ô¸ƒdEé;µêûódu/Ðø]…’,Zk1I<%YEÖkø£[‹Ø{¦'Å}bÙ29ó’w“‚-d¹È%âÛ»tÉËñJŒe˜bÝmUËî1&Ö'b¶]ìxÿ£ÐÖsÎ&'lV<®íˆ¡ü{闧!l¹µS®Ë'ßj—ˆßÎbÇرÂÐ:åæ­|@ t˜,@l0¬Œk™mâ¯oôîv¡äq^@{ý„!ìC‹ƒêj«”~Î 9ï ¸„aç(Ú@ˆyÌfìtö—6þFãÞC›|;§ÀÐdƒPå±Óìì€áŽ.vÐÎZ¤½£°ß2 1–HÖ~iLÐÚ?hþeãáj‹J‚‰·±¿ù,¤_:”×àkËc îÄãeŽS‰^W“£ ™¦¢žÀÉ£ D{æßÏSLÕ%¢ó¾¶Ôìt§òf—Ë.+jcgð 1egKlg´2Ûᚒn'‹f›¶ U‘ÞZ&ÞMÙ!aM‰¤*Yg]#ñÂë‚K‹a·04µÂ ‰Øk«Ö9ù·0­öˆ’­ °w1Î)%=|—ŠCã1˼<ÚB©ª‹kË5§Ô~®±ýË]yS¯¾ èË­z²PnZt06w…JŸ›ˆe”›æÿ3WŽVh6µ]s T…ám}ÖêÅA:–¦Â¾‹ŠÀ5vP‚Åéú§0„=—ÂP›^¥¡”ï=;¥ÖBFCÙӑ ¼äØ,TË¡PN×V¥°KÙ‰ŸÎƆßùú!¤jÍXtáhLöŸ×‡Œ¹Ò0>î¢kº`bÈ䈷™æw4æ€Ïþޖ·³9x{å²"5U)ÑÒp=åíía¸~Ù8Y®êÀ,;–f*³H»K:Yñ¿ëáxÄú©èç¶iû aÃUðW²ò˜ú±°=@àŸöTÐȋFÌZ±¦À¤c­…OÛ$ê ðÂ:VID¨àw±›ïR°{·Ô>Í +«UJʧÛÃ>C>ãQŠVP‰ +-în(I¹krçÿ”<—N­²\»þJ£âÕRF¥d1‹JÃa•Ãß·¨Ï?B^P3´æÑÝ9Cw£Lµß§Ï¨C¾¢NCüBe͏º™é%^Ïëh¸á{hT|¢qè6€ÆOìju¡WëÜðlµ^««oeó¸ö-óy¹‘ø†ˆ¥nÏ^©ÐË¢•>£©f;â̄žµ|‡û´ÿs%”‚RttEï”„÷¾ö'ÂݗZtúæbv¯}à`¼Ò•Ù ‰7‚´ö–ªjõ6×oÍ[Rɍ]„¤é#röí_’âþEoàdIÿö“ÆzµŠ²}îâ"¢ì  \Ô87™Ç»/ÜùŸ(üeÒ¨tß-¡²oÑZ­Ýç5í_,‘ê}c÷\JˆtñA¼äo¨X|VëÁâ²»>›dWQbÀ'ZšÔ¡*'Ô^)5’&Zۇаy¡Â¯ï=*Üy*Ráf·G1‘‰*m^ÃË9ÉÉ-*"ŸpØÒTäóf@qø$>œÁç´ªff›l‚ä*÷±h¬sõn÷‰» ÀK‰©T«0~ J#ö±˜îÇ°âðc† CÍû"ê䔎 ÖøQÐH8V¡õ_ —.}ƒ—õ<„aI-æ¸_‡ñc÷J²Qày¼Áë¸b§WÁçșÎ2_x›"ÒQF?è=3Œ²}¡„ŒbñI¹Ø¾È‚ã4¸FzûªÄ›^ ¿Ÿ²Û d¼0»/+,¾)kö\Îräcð‚£gö].EŒ°E { Aí BùÍÄ8ٟüØÀc[¶CÀ+›¤f Uàð'aXâŸðXèýK,\^°ÛàÉGà Ê¥u‹0c‹°<ãǞjVÎÜòjþX?ϒ¡ ý*É$x)ˆçšÓþ5¯y‘ó 9q>{bÿ^ÏJ×ù _ÂT’$µ˜PUcöéuËü`ñVØeÏø½Å62,žµSJ!מ ¯ñú2wŸûì4IùÇÐT2©_öOï-{Þ VP^÷²^^cs~§°ž° @“ÿ€½[‡( ìuɪ¿&ÞN„ü‚ß…È/Жù¹‚ÁއRº¨’_Zj§/1‚Òs0oó`¢,BÖ4y¶sŒúkâ/¤´ÓÌ'’~š)˜og<í«—h©(51tbÚ]zÈgê´ÝFcÛPK[šQöÓ 0 &;Í°W ð + 4ó­´0×T¨0›¹ßA3:E7ƒ*ͨŸê3¤Éu±ƒºÜ"þn©}4ù^‰õaÐhE$lÌ £Íöº˜\6é/4{Ÿqø‘º ~ ¿ôy„¥¿ ƒ±6K µìc† “9—$Gxëm‚Â[XáEƒÀ,Ö~”~‘é\ÕՂÔü¢ ^â€ètœ53Œ†ÕtŠÛ¹&,‡g|—:e¼S)Œç¹g’UƒuPlZÇJ¢.V)¸œÿ:ƒä¼\“Éâ*¤ßWƒœ†ˆr!¬îM¼UhVå*¯æÜ]E\aÒü’.éªÒèY&Si ÍæJÕZ%òï‚q›´5D’4¾_áhGÐlªœÁ’J¼$k¸&¢ØK”ÈJ2ØêK/^Þ +Yi,= o­žhU”šx3•3›¸Ï´$˜°³Ä%öávDº9 ŸÏ—v†ÁbYl¥ËÊR€ՀQaä¯ìžc·‚$éÖcÐFS\Fbr.äÏÀˆûNú~[;Yª§õJcV!¾‹: 4_ŠÄÆ#UG$ûªÇ¡8}ºNƒ¿p\U^RGøËðóò˜ËKp‰{ C9y_º€S„CChj~€¯4îóf"VHFA]·w ܚ 0Éí¨±ƒ ¿c`´’gÒ!!Z@P†ó;¸´üv}2ÊäbÛ~u¥ ª —d„cÀzýÊk‡ÊÑÆ«´1 6<àMb÷‰¾ A°ÊF&þ™¯u{Œxƒ ø-8’Äý1*}ØЫo"èöŒ‡tÙÖÁ<¤í-8·i„p±ß¥Á;½ôË7!7¤BD3€‡üU6pÐÝö½ :öWw³Á…8À½Â°ì7w°aÚ.¬!8!öîpeèX9‘òïUCÎ^ûýzˆi¦±Â1©îÛ¸BpÙ¬ø-¸÷ô+!ÎÏÌæZ{K—ÊÞ÷ø ù=ßRN6uÄ`“x—ö¿yH‘c±ËŠ?T“cNá¿#ˆ I8Æ¥Èڕ?JܕŽÓ~Ù®&ÇxÑmÒ·wØã˜MÇóòÿß¡N])Œ`•ŽÈYrB*»;H¬ý´ùòý-Pˆ_j?@ho"ÿ¦ã%'Ë lã6Â%¹UI<à˜´/’¬›ÿ¼#Ì©ìm‡yaN”#…ÓPÏÕ6®«×öªº¯4lyŠå7_ð®ý‰ g¡¡,ðÓ/vçá<*ú‰Þ}îžÆ´ œa]ßUõÔÒ´WhËôak#åLÁbNÒ¥qP¨–Ä8ZÓÝÂF+aÑ Áv¼ñ x‡“Cï?h„ð å|LŽwS ÅCæºTr™M +7>œÎ?ò6›Sê°8ôï™Kÿ»f.µSÒÒòbu§AKCøb‚ÈŸ­=äÓM®ÎÓÝp=Ïý)0l…‡×¯8Ül$åÞ?jëÜe¯w/gn>óÊó”~Vcw¦z³VeÏÂê®´wSï ‡»²©ìá-.­>ÀÀÖ ²pŠ-’ã²f9¥@&›Da“r¦³ÉG6Mߢ4Î^òo±³˜À÷_ÏÀ1¬§;™wv Y.’.nß0µÈÕIÂ)è~Ñ,®Æ˜AW8¨ ªß½ò'zù™ëä–=p-µ¸”ÀèR +<ôŸ³É±Bu 3*Kn‹þr¥"©™èƒ‡4õ8- õZAíÜ㍄{úl؁zˆ0é&M‡×a¶¥½˜NÂz.3éC *£!:~ˆ¡+ŒÎ¢i+oº[­!3Úë>ƤϛÌE\Ú]o`dQ…ÊPœ&õnö_:^z™zH© _”™Œ$•»»/²† +’ íC\½ùÚ;ù׊ûû* |$þ…¿joÂÒd¯`[(Ñ©[N +øî.¤|À¡òüŸ…ꯉ´ðØý¬ê]2èqˆO)Ô*Š”Êe³’Éñû¤jj l…Û4£•„ÎÔÒ¸´¦Ùÿaô….§Zzã¥ÂÅ}S¯—¼~Ãa¼Šò»a2šLn+‚~ƒýFÖWKþ™Mä„jãöj¾¾2§ +uVnüñB&ç´$SA®¿­(ÄMN¡žëE9Ï?Ôqqä\PVõâøç·(XE¦z÷—åkDxk)™ºÈD¤ûúc‹´²è‚°ØLsÀÚEO#¢MJéš ½ÚX ‚?Äë56#ÈeôàròàxºNŸ>š"µz 2ñóÂ(ù¼Ž~Xo»ólº;îßizhåõ¡dîî¸Z…ÕgD¥(zJg?t~öG+S(PïϾ¶N‡Þ/…+Œeè1"«Œ°xY Ú Œ#<ŒOKº[¼(„?ðyýÙ¥Æ89¶Ù]‰ÆÚ +Ó¿Æó…Ú{™öÕ]®q£±^ª±ëx¬TÌҋŽZ™§CÀûb± ]Åн$‹¶óW,Æ8—kUêr +Mš^H—èaEDÍî“Ô‡F½“µÁ«nû÷V]#õ' eMŒiԒNµ¾,Õ@±ý7‘­1,ÔrÂ"Ð9cÜ®ùzñ‹uŠ,äâ[7Èê.ɶn…ê;Äö·È^µ¿ÐÜç#"’3šj‘?Ì:w­ŒÔË +цSRep©™šaÚU)ߎÍ+!©ócFiû/-1} »®¥þÞ\bO¹öÖ÷\Λh9ïˆ;)¼—ÆÛ»¬¾ž ¦aËzVžnâÆÞÖÐö1Ø͂¹è>èòzÎú‚8«Uz}µ±5Pÿ_p<› +oz]PĸPT8„<§•q¼þá;@}Ï.X7Û:Ån_;Ͷ]i»¯ýz[+”9mü‹¶q5 ´ºÀDN€àFSy½ˆ¯ÞÆ– Ƙ>ð0÷÷²þ7ê²/>ÿÄcLF.ãßy ¤üù7Fx¬…C'eÒµà:[‹³ø‡£ÌCº\rë!] ®¹…}Ì­à:[è}·‚ë@n“Ï2”Áu ·÷‚ë@nµ}̕à:[m¾8\ĶÁu ·Àd·‚ë@n=¤kÁu ·Àä“x‡¿äö.+˜r‹6¥Sx{¹EÛø)¼ƒ½ÜÂ)'ñörë!m·ã¥O³ßB¿¸\r‹ºÒ¥à:[ÜûîׁÜzHׂë@naŠ¹\rkñ/¿\rë!w‚ùw£Ìs"ï`¬eûé(kÒ´Sd µüÉ(ƒö‰~4Ê<äi¼ƒý(C^ìÞÁ~”A.'ñÇF™ÈÒüx”áæÞÁ~”¡ÝòÞÁ~”¡\þ£½º÷’ç¡ðpŒ–UJȒ"” *â¢(¢¼àâSqÿßI:e´TÚúC›&'ONžáDv0—2|÷ͲÃEb9:•É®ï`›ðGd=„-›ðGdáôíØ“0ð‘{M‚‡°eþhàVÚ± 4 à”vØ9ª8˜Ê€ÇŒ6á”rEʌ*枔!kÃ&üÑ$x[6A“2ÚV*óz.sOÊd})¹,e؍+¹Ì )ã™×»@±¥ÙÁœ Þò¨ì`N +JÌI!vPvC +²ŠŸÌIù1'²ƒ9)h*vdvÐý-nñ Rðæ×R€s1u¸Î‚ÚÉû®3¤€\߶ÃEÊØÊæÈj«™̑õÎdsdA_ŽÈÌk;AàoÁ[:’Ìý­‡p&;˜û[…ùÎæþ’ÅùõÛRæ±p¸ÎHb~3‡ëŒ”a†1q¸Îø[aîp‘2¸/¦×)CÒ=)ór™{R¦û17¥LOKi_ʐŠI¸Ð-d±c ÷ºE|$á@·ˆ-Èæt‹ï‹ÙÁœnÕ,vlv0§[XŔpIex/6¯}d=„ýÄkYÙ)ÙK¼»¥ìeÜ+eBÏeîI™‡Ðs™{R&3?ﲔé®IÙõÊ)3¨˜‹R†î‹šËܓ2ÅÁږ2ږ”yg²ƒ¹”a}q ;˜Kî1²Ã>)ã1²h»‰×>²°ŠíÄkYp¶¯}dá¾Ø± ”2Ýõ¹)e bvlÂ¥ 9%6ÁBÊ^Û z—”Á*ڮݓ2t_4ZwMÊP'«¹Ì=)“]_Ì(e7)Úi)ÛT1w¤Löc/.Kêd5—Ù•²ÂÁ©L^å(Â=€n±c ÷º•µò(µ¤Û¯B̶ѳŠŠ9mÀ6zVÍbv-ŒU! õ½ ¥”å{Œ_c÷ì¶y¡ÄŽ)Au™x^º6È×ñIzÇ ™zàSE–£r:ˆj%µž +'O¸rì“áÉH¬„îþŒHêt‡J½il{²»4º4÷ƕŸg58«ÞrëE¤U„Žð…™É4+0¯·ó&ö–u**H÷1$y•Öo;@Š žäë|AØ{™‡qbL˜‡Æ’†>É®<•º¡ÔÐÑÜE°ÐðAÉ-Pêóuýz¥>ƒuMVBÍâÓˋ +Ep8«1“å4ÿòY¦"ú¸ã¡Ì' †L×Ð|µv}ÃO–_r°ö4-pq7쟥àÅûšy þPÚÿ’Ü`EÀåZS~ôEN# #MW,õ–ga\Žd^³¡$4Ã( +žÌ=dpÁdOgÜqŽÿ ÿË©“-"ø3}‹v¾€k^*AQƒÁ~ž`©ìå©.K0_ _õ¡<ÒU¬j +¶Õ!ÿð'J›ÂÑfU¢8ÿŽŸ§')>[†ÏÞglDŸ!æçۃ…¹YjHn”ÄU¸7ñ÷z‡íå[åHú÷g[eþ†ÀCȅƦj”£êa£ÓjJéÒ¿gÜ ÿŠ£$X-¶¡ù±£à +[ÈÎÅÂïõ@Ŷ[Œo¢®W…àßÛHê^;n_ŠÇªæ뱔ø•Ø85äÇÖÅ}ž+ä‰mzuýmò萳­_êu²ù×ÆÇL#ýSt!Üp̋˜2€ŽMž<-$yÒ,Å?jOmäú’ÀIíÚKáSöÊ4&D÷œí-\Å?¨†HH°×¸ðWsÀaÔç¡lòÉ#޼ѕF¥òÜËb’Uؒ™¼ »U”d.'ÀÓÉáJHÈ=Â`òG{¹@u!cQ*Ú¤¨Kˆáq|áֻƺEe•³7JÙ.$ñcUÈJøÑ ‘õe³0w#V/eG!´e!@ÁZ!²îo—‚ø×¢ÚÚõ·å»‚÷²wУ%0 M–CFKt.ƒ&+Ë!ÿy͇ ½tî,·Þ‘f–Cç{‡€ƒU -ÑéŒ,ÑéL–»†ô z·D«!³Mqm Zø¬æ‘-”†sëN–F–JK¥÷¥u'ƒ²œçÛ@iáµîdim `ßï·îäÑh*óŒ&–ŽÞ÷¨uòhº³Ñ ¹ê{"ö{Ï#1Ûþ¥<1“·l¤ôYÅ~š™|N¾*ÿ¦?àÒD8~~½J€¨eñ¦q`þf” žN¹RO¢ ,‡]9öʒb¼û +M*;s2ùîÉ»7–Žµ½>±èCù$¥›\±ßÐ|;P^ÔDJf~œOe PÒVÄPz«0ÿK·¤j@³€;¹s (û„F™.íECΑ.iýžÜæ³bñ–Ì}Ÿå˜»€² ++ùßhLDýЯ¢PÂħŠÕ_í¡;àHfwŸ·‰‹bj0Énd­íU‰ôÀiõ M'^µò»Ø·€(܀S/àß̱½9ØÿûG֓ÄZUü´Òlæä˜|õÚ7O^r¯qO£x i9vs£iÛ͍£!ӑP]‰5@­Ä±¨EƸzÙéÐs—ŠÕçÙvö½K»;wõ•ØûÞ³Ö©íéT£öj&v¯71¯6ìEå½Öº¨ µšÜ²z}ÉøDÑúxá¯åéRF:ùrþߤŒ'Ýf·j”?–2ã¸6p«j%Ü›ÿ”·ä„wvß˗·®¿Y¾,¸u1Žk·Øcv{ùÉl® ܺÇÀµ[èd§Àµ[ºY8î܆,¸§ä¸6p«î•N€k·Ä)9® ÜZn.\fáýø[¬þ>v‡í¸Ew±Ýa»¿…Œ9® ܺÇÀµ[Ð}§Àµ[mq\¸%Ln,ôk·.çÀµ[ã¸6p«rìÀ­}ü>³ b{Ù¶gª¿—Ýa{fM{å¾¥ÌkHx˜½ìÛ¥ ™¼Ýa»”‘©ÜÃî°]Êà[¶íú>ÇR†v»Ãv)SýØKuJŽw«³ô¬‹ùf©dì/ ˜¥gqö-]Ûôù"Ë×¢©~"ê·L¤1¿`øÁµ±Ç­œÖâ‘O|v¿ÖÞ҉Êá ™‚ÔfßÔóÊ3=·X¡ +gŸSàm$±or9˜‘ÂGbp¦Ä…GåE¦¹ÃWê¢!ò*“O¾ CCbk‘¯H­øï?$À¿„f‘FÓÄP¾ÉÁ冗ØÃa¹Ïc€ŽÞÌËÃLI yM’~—Ç~Já¨ä¥Ü@®µ<¾}Õ4—ø8<T:÷fcZ&ÚãÏ®1\ElñIÚ;Ø,úñªò¡I…Ÿ'oU©ÈŽcáçtJê~f@ÒÞBg_0%å”{´î +/)Hj\¦¿¢^9Ӊ$á/ t_öJhªÞg| 'aƒôÜ?íë“u¡þUyÃÏ\,)ÔÎoŽ4¶¾‡Oã“(Œ™X5Ý÷ÝB†”Ö:Ë#J쑒\0f'û8õŠútÓÇäì]±øí>ñ@1^æ4þ XOñìQҖEžJfOAïKå­êʛ¼~ªX¥”^û‚‡¤þà)™•¥¥¯wý2_ÿ׸|¨|•I챚Ķ2à`¡ýf1±œ/Ë^¹RƒâĐÛNQ,ÿ‹Áe™…9ÁnKÔÁÀ-фÌL}’ìMfŠP¦jNö²çq©è>Ê +õÂa:GTÄû£·‚Pk–€0¤aM#® ä4÷ úÄý³ò©ü5™óæΝH,¼`Ýý¡»Ï8¸o´"úâñ×`bvºÏ½Ïxø°:û©9c&Ò9´õ Aã H˜GI­Œ£lQ1Ì‰Ï Ôh%`j:Z÷ñ¨ª`u)þéûàcýiÃ.FPq1̏™ÈYáÂ`²¦[<šê”Äò0Äë¸æÖIlnœÁ´âi#%Y=öOE—Ý.%SÂØw,:€Ð&g¦ü{÷uJÉbäƒÄsû毛{zi»ˆ_­ÍtùÄ£wὌù2îtÐÒ¶¶ øڅê4åÖÜLO,¥Ó8Ñý¶dò&ì×t¦{ÎlZ*)´]ݐêWU‘î…ûó(þ%SÇAÀòa”˜dL-™ZÁÑC~é¾õ£­OS5#¦^ w•UTäAÄoÄaÊy-Ç7ËMδò×Þú÷Î ó[oý{gM{ì'où0w2PWûq.„SýWÞä­&ºIÛ+Q_Ä@nZ0 ‘2dúz]ºæ®­úŽ‘µ,<Íù̯•Ø8X¿˜´ð¨Ƃ‰AZ#}ãÛèëÐð­î>ʹ·üöC žEª²*‰® $gˆt2NÚk&°™O£6'ˆ=Ëây +îÌóÔ8­YÛ}™öA,1ÆX >=L%l|ë–~ú¨´NäìââÜj?}™@YöÆÇ°#ο6Ù8ÌØ7kdq.V[‚t>#§EŒœÖL ›F4-÷½c‘ËwÔP­¬O.¶Däü .Ã:LàŠx4±%¿2%¸¿ì´%˜„6¾'ÛMIklÑO …Ł’x’‘f£(ró¼€:ҝ5äç|Oì&éùQ“´Q•ÔÓbqâKgïÏâO†´õ |òºþ|ƒE里tCQÿÒv1˜îYI/²a4%êÆu9j,‹Hág(HfH;óJ0 ¦³ +-Þ¯ò84϶³ï]*V¹ëÀø ìÐgMxÚ³ðdb÷º"ñªú\äÑëtÃÛdö» Á“[V¥Ê¾tÙíýVäïþî»Z ¼¤Vœ†ƒK‘¼ã^I”}v"Y†ù}[‘6%¨€Ò>ä¬Â›½È_êU÷%`-UY,Ò©I!¼lò•›[ã T?ؒàݖKËÀ¿Ä^4R:ôúÅÿe/¿Ÿ‹ƒiSÀŠÁâU¯B„Gºýž¨ÂÓGˆð¸*=•ñ*¡iÒ9O5'Áv0é„Õϝôì1VMªÞɁì¤hrŒú· âHxŒÒ<º%­Ž+]n¾ÑùDஉC³oªËs‹ªÉ…g×æ….gGË ¡ œó^”t\‘3&Ÿ€}rÅP»ã\ëêl=À‰Œœ +?Ðy©cì Á"éNJèåƒEt@p"§—«ˆ4š&ªÐ*ï˜Þ°_b‡IºZ5îù«‚ž_ñ'L­Fo6¦# ™×5YnñÉÍA1*Þ_×ÛÀƒ÷°™*ò?| îK&Ýáj…8}ŸÆ' xO¾Nâiµ¢FÃK‰t¥|x\¬ÉÞ³Jە¹O܅X‰ñ hírT,wZ°Ø2|ƒJpÐØ1|é !ÿµtÞ³dQ°h ì_´‹‰€5ↀ‘Ñ™ |&íؗ`ЂwV"j$‹h°(Ì7µNîvo aŸIãÃÑXÀ0B8ñë/³¡>s}¬éªkh%B_ZÅax®ÐdƒPü$1ü! +8:ù ÈÈõ®‘–}Þ¯Ÿ"÷½ðÎ@I¸=få*²I„K‹±!BŽ%ÈÅØ!Ǥz~¹u"A.Ɔ9– ºWî!Ç„Ù-B;$(ã/rª³Nõ/Í8nô‰é"ßbY]$Ž¸:á×ö oT*ð/‚ìuóUË´} Òfÿw{Pg^³ìAÛ· מö úØÿžÅi endstream endobj 24 0 obj <>stream +H‰ì—i_ª\Æ?ÂPAـ榀šZ9–Ú1‡Ô²ŽejßÿY{jÃñF±wÏ´‡5ü¯k¥ª‰¤œn¬rœÉÿUãùfz š¢Ñ¸¢©2#âhrÿlȌ‘-sjû6ˆŠyQá´`@“Øt /õ/ç**ô}3uîµ¹ôÍP@wYµ¾åz¬˜oÞ«hÄ´uò·ˆGóÏÉ~ ;¶41}Îá¿feoÒ(ʌç¬$L•¡„ŠírJ,DQKKLT҃÷º8¼ŸÉð +҄i^KIþN­‹õV°Ô¬#”Ã/‹’ïñÎó÷¡Ü¥¤=§¥þí¤*e[«,qFÅJN#×@ãA/NU—}[^áogœZafrºéù@B$ÄÂý*Úúj,ìWy™ˆÈ©ÊÛ ^ÀGÑ\úi%ã 4ð:,ÁæÈß` ї3¤ËücÁÏ©§)F˜VʆÈÏ5„&ì{H˜ªJ•|4/òì“ßZʟšãÖÂË0_#a[ïòíQ5¼”–—°þN_÷!%½±¶Xœ³jæá*öêsëÁ –µ¾õ‹é/ÿ±ôtY¢%V=ÑáÑ7œÓ°v9a}“½U’<Ñøó’³Î€Ÿ~Mtɟ>Oâ Hýf#«Šwœïv†ÏÙ¿±ÓÞåt±–Šöð½QÃÃ"¡­C©¹¿hÄÞAö+!#4½ù0Ãk¦®ÝDä˜r6&ÇΚJBË4™d_{Š–æ™Fæõß>ËÞîs=ÿ]Iò?´¦çɲg§m„9§?r¾ÂŅ¤§£öƒÛ y ½ª ø'|äKø~·ˆS‡]³ìĖ#ðÔÃPô·$Û ËÈ;I2¿ÎCr…&b9UHäÍ; /'!cÙŸ­H‘âÿå(R;Λ}(Ø31VÇò¹LU»ÓKžºÄ*Ja±P•¡žæê ]+üén}ÎûHàµéϵõ­²1µ¬|Ï`ì1SÿðNÎ¨‚pEθi2¥ÛÛ¾Ôý¸?צA„¼ ™Ù[y‰  ¦WqëÁðšÔ ý}…Þ†MÁºð°…pù­°®+9ëJŽlî"Nò¼µ} Z®ËŸi!AWª«4Gz J»n†ù½1$UNڛšWãï³Y‰'p&µùíö‘˜3‘{‰²2@†ðôFG å¬ÞNj¿/vÃMs»½sÿ-¹j-¿=”2P'~4~I¨hҜq_ñï1øÏz°¥¸Aļx­À{ZЄ¾ ’Éš¹˜}!à]¤8#•µn…e!åÓ?~¡Í¦2=b54é¡Øç[x"”덄Ä aÙPÎ" <Ñh’‰8ñ×°ßãµI~€~3;^üYš2қyTH„_„3D3DFP)1RQq5º€z©H"?Јˆ@^ðbrúæªÂ/àJBlµØgíR¥€çVÁY-|Ò®û ž×cžˆKIÑöá-zAq^/ò…s “ΤÃõ~åm«¦G¾Z›Ð —/ìçÄ6ºŽ+9Ròm3Åú_r´õûI…³i7ZäâgÕ¡ðÐ-ö3…³I…¢Ï'M”M_)åh†ŸÓj³Qç=ôŸÀÌØ+ÙÒ=Áq©áá:¾h±n/#òEî²KÀøGE]ß%+eۜA³ûÊd1acv‚†¬ðPr¡ ì:ôƒ´¼ž'э—ìBù;ùõN®ãût(ßÌÊ4Ñ Õ ·>Ì4™7m ­Þ…ÓeŠk&gj3|•‰vG>óöþNÝCÑÆÃË{Nv'­áڇžgm¸–ù-êš×ÖB&p?ºßü68¹ ZžÉæ=¼,¾’†®Üö¶"+µ³§\Ž{M]\gƒ´\²ÀÇLF¶V¶VŒW>m"kõ¾ƒØº‰,f²³Øº‰,QdG±uYØå_±Å ñ<Æ0o„~ºsâñ•mæRFøìçaçMÆÃ?=7`çØjãIWÊ·ªp†Þþ´ôµå‰Ûô“ 9gÇ ýb¤xl{Ömgݐ6V »‹ÑËÚlñÛ.¬ „ðßB–ë\ˆ¦ÕÔ®#ø›læ×V{ßOl#?¡÷·£ÔÞ±ÒÆ,DãÁ`ÄHØÆÂÈÁ—9„¾Çã íÿY +—ppã½~r^}lïõ“ó¢èc{¯Ÿœ—©b[Þk3:˜œÎ 8/ÖÄ@ï?18(WƒÓy´ÒÅÄàt^°9¶‡”Eþ?‹ýÎ,6™®kv{؄¥Œ0Ù¹M8PÊÌ®tl”2s~qlŒ,¹Ëñ ؗÈRô¯°/‘µý˜[†&åpèŸ&¢÷² šŠÞË&h(z/›p IÀ]¹± Ýf·M8Ð$PûلMî—=l&p̹M°MBšÝKÊ(z/›p ”}qn”2RcÎmÂÞR$¸…ì»®Ü∹®ÜbEv ÜðہRFòò_ÀyöÉçf*Ûr}‡×na·Àu€[ÒûÛúêGÜâ]\×n)Ú5pà–T²;à:À­©/_€Ë{*ƒ»ev؍[Š>Îì°·¤’îfv؍[k׳Ãn܂ƒ=Êì°·í¸p‹kìyì +¸pK¹®ÜR´kà:À-®1§À=ÜßR´kà:À­5‹¹®Üb&»îwÜ^οD–ôþf‡Ý‘5çJ÷³ÃîÈÚäÿ])Ãyùyv8¦”aŸ|ŒÙa·”}òc¿&eí~vðÆ_„ÝRFÑǙvKEgvØ-e؍oÍâë¯H®d34¹?G¿'eԑf‡ÝR¶öc¿*e&Çö˜º4»¿”Šev؍[¨±£Ì»q»5WWÊpÖ5kïrdö¥fÁ)ý†ûR³ØýW՚ü5…4‡]ß%?‹3sûm‚ÔÏI ©ñ‚÷Ùb2a•ÇÄÓÄ@v¤ošuÒíuÍßnGö‚M¢ñ ÇowÉ –•™T6'L•¡$ò=VGã¡¢JÙN¼h¹qX¬µüº˜ÔËÃKra³s2[»©0³/â…ö‡Îø¡ä4UfúÝ$ô•Â‹yñÜ N ZX'0?M1´R6D~®‘Pú„©ª$PÉGwHµÙ‘ ™lÇÉbvòˆD‘ôì:M-è}(‚îÀ«u{åìÚiV€ Ú:ž\ˆƒ¨µR’0ÊPµ¯‚ä÷Þ§HLpÅßpj{áùGEäÒR’A“„/ iÒ\ÉRÄ+Òc +xQsê*ÉYJN7?š4"<Ðæ2“¼‹Ã“òŠÌ,iø;þTfàÆÆ~¯O /ŸÊÀd|eÃ>X1‡_µÐZiț§ÿ£½:÷‚øä!ZBšºx¢ˆ‚ E°q +*¼ÿÍlzDŠ/ù¡ÙìÎNù­K´öªô!©~%çþìQÜS ò?z珒ƒ8ðX7g_ûPåÝ{³·‰¬ãÃQBÞWÔÃ¥Ÿí˅»rي:£€Îۏ»Wº6{žÄò®à±AòÎàßî§ E­“—íý¨TVî=JxʎûeÕïwáh¡b©U¸ ™i%ödG+0Gªˆ(Óï2€UÔ«K<þÒr`–B·ó‹!'oÜê‘n$¥uŠÜ²ÎÜbã˚@"o× Åò %F†´ÿ‘@ц/iO‘ê¢ÎÍN¾ËïsR+îû‹’$ªDø£ö,&õ‡©º¼Ÿó ;®9MÁfs9< ˆ"ÏWdJL¸BŸãv7oý†…58³­$|3GFÁð/߁Znñ¡Õ¼T–ÿ.ÔüùP’‘éy]У1z2¦ßD⠛ȍãë?ük¸žáy*¶@EDZY°‡òLÍnW¸ßÖU¸F/´ÜLËLùð˜\Ò)Vߥá@e–PÃÅ@¨¾¡‰ËÇá‘j‚s‚†;z‚@ƧèÚWU¿HWS­dä!ÃFª"L‚6C&61Íl0R>ïÖz®¡}ø€ÍRññDØéør± ¹ÙbºYðâ;:5¿(fY_䟜×Ð _àì[BÕDÛ7±Ò –6Aldþ`¢÷K÷êÓ>¸=.h ®UxRv]-"8Û@10d°±¾ +¤ ‘%C£Ããx/Š Ö$5âpôB[MBž)Èp|KIÈ3—´Š„ )¨ +1SHe1á±y« +VE# 8Åj:£ªò‹q5˜éxk‚} ’ÌâÁ%ă])KGIlºP$çÏ0@‰Í2Ф*(ÿ’Ë*ÂafÑM†–Ó͂'„Èß©ŸŸ9u(_4SLeÒe§…‡“b±’òþËR1wÀ=e +ìäµïcbõo¿”¼õE¢5_$7ná¯_øéóÄå¡,u±„dVz(ȝuœ’EÚRßֺЗï&€ãôÃæ»H †ŒÄAëÎâ"ی°0C}Ǟ3ðçú‹?º#¤ø"Í0*ZSj$ORÉj¥p¢^•ë¾6œ’­A³ƒR>Ò>+©µë+$T!ARiœÚù8µn*…Nš0/ß=ùThÏ÷_ʗր*š*åNº™ßè´Õ»Ùh«v?K|vÌi|.$ÞáϪ P]€«ÒÃÀM¹WZ­="öjÜÑÿpRƒ¸_ +ƱΡ†¹ÿàˆ.QúL¢ó·?]¹õ’Y¨Ëš¹õ’YŠ^7·^2KÑëæö‡Ìfè5ôEoѤþ¨(z‹&õG} ¹$C!TLõ¿EUKÑD×BªQK×n[ÕR uí¶U-tò]»mU«ó‹K×þ¤j#ȋà¶^G!¼stѝ•§÷@§’)Ç&ä<_`Úa(ÚÇ>¿žÁÙəƒs«ÿ€ã»'p%÷§E™ s*ÿ’HÂã¹PFƒlN*^¦ZC²Ý/k3m+énR…%rÍîÅÒÅÓs`óæz•ªõæ*@X!-§ædF®àƳ¬X³p*§;åþ ÿ´še‘ª$•Aþ%—U„à }I2vþad‚Ô²§mf +,’™5ËÔÅPpÿšÃU1:§Ó´ø¤­à cb'->\¶ kßy)´›!9Áž?äzïs‘}ʊ‘¼”f„G%ƒ2•xø3¾/•¥§ T¨‹äæ¬_*§ß2r¾“ E„êTfÒW)2/é8Y$3Ÿá<ü™üÔzýp_¶a–7,öË4F“µvLBð³®`2 Y燔Ϫˆ–¤½*}H«?Vtø(î)ù½óGɑ<$ÖmŸ=O"9×g"Ûy²‡uñËÀöŠzè8/«~;OŠÎ@¿}¶à<&¨>{ ƒ$¹¡ã<•wî½'y®â¼_f­»H¥ùUÁùaVÜvYcåÓñÚ«l-q` ?ï0A ِ5¬‚‚( +¢( +.ƒ€lßw®Î³ŸªîӌǛNÕݵý‘ +¤\ž< ¿Æ§ô­pSœ9‚DàL)!DÀDw!Œ­!wåqI/õ€¨Öït8óŸA-Ž¢áìi¸¨7Š&Ù¨ÊiÕ]Â'¾¼^>ÐÊ`EÞn™ †¡”L>¢=ÝqY`Š¿M–ÉQ%àtꎾ·>×4AQicJ%ßy‡"ŠI¤À!¹1@cÒ òµÆœÕY—ïß­£´2ÕÔúP½»%¡95\$)õ`L?:ˆÇu@ìŽöÑQ샄¶$¹üè ÞÍ×âåŸ #q‹ +¤®´¡‚9¶¡ämvL%´M¥Q _6•Zqu› •_«èràE *ˆ/Ÿ*i“-ÞÑåØ2¸MÅä%¿M ˆÜ6•–þ‰#rӞ,Š« {4üSö֑gnÁ)>`šÒB³CÆ5̍€6ÅSàò÷cܾs«Ç‰©*8À¼{wȳø€¾ ƒÖ ÂæqHÀ@4°çP½ïßÐA2ìD)Kù€úâRe`L\ÄÞ:Û­‚ì.PW+½XEe™¤)¢²bÿQµÙê ™¬[ME›­Þ‚Û„¸%ˆ^èŏìY,•qf±[#®¥,AQq \œ8UӁ{ûRœ n¢³3òäRF²Õ{doK½ràÀÖ¾ÕTí8Êãßêö FÝ1ô•RË@ÛA‡ƒk*4'T‘!“)oTBÖ¬² +2~ب)D‰Ñ®FSò Ãc€Ë65 ôe¢Ðø‰>EdL¾S‘餴0a{¬SW*–öÄ'óe¿XϽ= -2bŒÜÊ~8yöšêëû?XliŽÅýV2nf[þĽùrP}Ï^d'W¤óƒÏuƒ=š¿âcûÕbãagrÇ÷¡&»†¡_÷¦B§ ÂM^¤o}ï%Œ<<8ý¿N~ɞ„GÍVt½5L‹óáðbøŸe~ú¸'Ë_ñ_jöܬT£ùáãt0u ÜýÝÃ%v4V?a[pf¿öì’Ë?ÅJ£ò^3ÙÏ?i”«ì’Íý.ð~1ãä—ßeÏI +ÏÉwƒ.'ÅsN(/d\v”p-Æ1ø±T+ݎÓÖtÜËç¹UA:©]å«îܒæGŠºˆ–c÷¥$„è*7lfÔÅóQü¶qË_çÓ/‰´~¨äıÿܼ­¼å Ý{©OZzúŒnjÁfì³ë² o‹+¸ \©Íɗ½'Ÿ\µ9!Ô÷ÂFÇøHp¡l¸Á…T_ßR\hrvƒ·áMù¯ÁҁJÿs`é@%!€?–TB³ù1°t ’K*~ ,¨d»üX:ö<üρ¥•É?–T"ÔüX:PiәÏÀ’öh$ÊXÝ©tw:ʐ©Ðj3ñ¯Û tl4›mF<½ ¯ Í¬â ì5eNÓ´ŸmÖOÛ ÍødG Ð×š‰yiUüù;SêÏy:sHˆ ©KA!³©u]l.€“Úð÷PÂ¥ˆÊX{ +Ìȝ) XÀd¸[}ì0­†L‚£¾¶— +…ßW9.a¬.¾Dáø”CAúò´JÒ~Áv9€ŠNˆîÃj5•”¬#Weã-ïR…FypY6ïGف›A×±‘\w!y)7ۇà<× Vží"W¹ÉZXnÊô[ß[y‡eE¡1ß%#‚PJJŠÄ…aÛ:ïjZQ­-æsc‘H®(0ù£Ó"îRw•×ÌzáoÍ%*_‚òӑcÀÃãWÑmB~-l˜L_º@NF@Öb°3Ȓo^o*]Òü$?Ÿ'§A7|ß M¿ð‹¼cBz£ê†‰v»‘µM¼ +®“¤.r-¿ÂM"ö'º:ø8/#G”:ÒD¾"-Tè×ÛÅ]˜à¢•wË5ÏïèýŽŒç3(‹öj>ˆ·X9^*H<ÖO4µ>ô¢ C|çÙÄn<üK0sD:á-ÌW÷Iƒ°e"…t÷)òÎÍk¤‰—Gù‰@dÑõãðW¼ãpÛ6ŠöØyÞZ§®à¤:ò„y,'V  ®¼U¼¿[à'©›٘ڧysüÃ4@‰9!¿8·pãvª[·Ð^K Éå‚î<ü`¹ ®;.ðð¢y=èYN€ó¸€¥û‰,}.¤ÍÙÔ傴Ÿ@&³m©XH¶;Àa¤ÒMÃõs±Ò?ʬ+l'¨Nb·¹©Z N˜ç:]Ç Âƒ¯yóIˆ‚Ë‹üwqù •¾4!n˜ØÁf²ÛDðïM¬>ª—]+B”ÿñ 6»Cß½†òiY“οË5Ôt~QûnRé_ðð[MÈâÕýUmËAÿ]ßï0²ü‰‰]3B¦)Å<ö©‰µ3lšPvg8s&¤'£‰~º¾nâ[I…œ”ý› }W:•Ajӄa™ØßsÊká-UÍ5=5èèù&áDƒéÇƺž´SŽ}u+u—®ãá¿4úfž«Š+Çþ­â՝Äα?Lhß½†«ëxø»†ñWg 9¶yŠðöj+G’'¦DÏ0OEÃkŒÍ®ã ò®ž0ä¿?Ã"ßM*Cù(#þ@䯢aìBnHãÚh["Ú®yùÙôõ¥±•F˜ 0%jfü0G§2Âô~(VéJ S„3Ózñ#6öՇ“ÂãhÓÔö(dܕ²¦ög!œN}F‰—þ[!3H8ÝòÒé~ž¾ ü:úhD$K`ž™),ãY„ҜÁ2¢ËÛŐ Š°ïèè¾x'v«p—§^æ9“¨qs>Ý•µÉ½-º&ÿ` d· ¥lâ “åçä“-Ò\ÿ ù*/–àT¤ÓV©æíÛß$—2LX¶ ä¼Ç5ғO!wRò)Ððõò~ c~jPÛ=Þ¶}&Q•‡wÓþ,dÓkŸH eü¦P•Ç}­œá.ÈÿõÇr|`f¯»º6l(0@tŠ’zr>Â~p!©!îÇmÇ\(tMôjÆÛ9_cb/“Ð.}$sœÆ0¦œ7!‹ª~v+ûa†‚ÿ£½J»šJ‚è/xÿ¡%‚ÞQ4,H‘ÅÄ@ž NH0 £|™ß>·ú%D<&êã³×ꪺ·ª«7M_ß}M«cH¼M^öM‰âšDYý½ÿxf{®çæ·Í˚?hM'²«Êþ֝!÷Þ©ÚY·Þ¯­íì^ÑKU.ZxñLß¹Fµ´ò„RÅNÁnµ´¿*ñÕÒñ†¶ÚÛjغx÷'i¸¨–+ÍÆ°¥N, °I⥱ÿŒ$îãs¤ºªÑ®‹bwãbs(û°²[²m_]Óæp®õ¡œ03ËowWȖ£Ò–}¼ÞêžÐRä^Ê0⅝šÂûmo:€¬¥ˆ0¢áÖùp ݽXb¦Eu0Mº6ϗÕz@kåþÌü`þ£È?äEu±RIÌï亂aŸ½.ö³;ÉÁâý"ŸEœ¿šUW«§»NŠÜ«žÎ>šIÙIߝ[~QèP~ÒD6¥Ÿ¾}¹V°¶üÐÔ6Nñ=Q÷•Õä6\3ÿdf¤òZÁn +{²e ¡çß®,þ]k©…ýí§)ˁÉÕùºöhí)-’×£vá£3_ŸlP|>(í«i”ò«¨þ±8[Œ5[=Gw•–¼¢º­ÿï|6—MÕV¥l,wZoòσ¥îñÅYÞdSSÙÊç´À6ò“ÓN½y™÷2ɊŸÀ¾>2©SÖ¢ci´~”M§µLVX¯–FµÖ,N»fï’=¢¡Ýõúöê{Ċµ ¬cÓÐF4°S6‹uа‘ VÿÝOøld‚ üiœýíFñÛmfb¨)v_¢óúÄ [gïk‘àÍÌ1iØYf¡F=5”·h¹ô»XQÏÞcý&t°J¥q¢Q.ú@ áƒ7¤ƒS^Á!£“©ƒ )«¤u‘íÖ2ÞIŒ +ÆX.³LÆÈ-ô0Aòè ”qš k<³»™†«`3Aq¡”eNa§gǙÑS^ .t”L«À½‰Ì9î¤WLÉT¼rœ)g¸‘¶i݅A0‹Ç¨=Tä-̇IØ̔Ö\E˜xœié°Êo\¡9¤peŽ$e`7Œ:ÎúY•ÈFD«>ïåùv§ÕЩ×<ËW;­ü3ú’U·ÝÞå¨?—e„Ú1`Ö"Ó¶»3$M‡ î]”îVv$ÆÂ÷R°õLÂOVY Kí¹Á¥®ô[œ.:As-a4u³­!Ç>­™:Ìîb€w¬KËPh°êjސ¯Ú77\ñnb3úäI`S_ÃFæߘt&’%B;bra‚–F“ Üџ§%`¤¤# ȊJK.#ä€”„TŽÄ%h pDX\‡mÜ3‡°E +:e4šh¢Hn“ð\R_i—æÛ7Öhã»N|íºn_–þ®¬UXà%à " +7Zσ×ä.µõhp/"R€/ ̏P)ISVRÄcÀ°”§yˆŠ ¹:ÏråÎí¯7N†Šù…¨È[’Ú/‰ +]Yð’2<]d*Rv÷GäÔR5á‚-/‰KAGAȌoŠ› ÎK\q­ÄØ×´øŒ’=Žì(/\EÜÓåSŸHQ?–¢ãË ÿ„ZºÞ=@²"á7!·9®÷Ì ŒaÔ!‘k-$3 +E®f&%î| ¾ÿp¦CCç+Y” ‰Ñtu ύžb ghå (q$Ý×wMWqL?ŽMøEÍ5’Ç,ÂAJÊ7ÞOÈyKJþy–ráÇy(ì"U˜©à!ø}‰tV@ +BªfHV,`Җ®*àSc$¿ýÅúÉ@³c€6¶æ„Mâ’3E:/ù%P7š"Ÿ IºË„#ß¾¹ô§)³áj–Æ‘ ´CI¥/Õ3-\’Q€훻R5IÐh-Ó~”çÅ6N®EÌ S Zj‚ØAҍtp}¡xeá͕ÞBâöªy’¿é5OÛxsô›ÿä¬ÙétÍA~ŽvÒËûÀ"gý¿ºŸh[FËñzÛxžý'À=6©& endstream endobj 7 0 obj [6 0 R] endobj 25 0 obj <> endobj xref 0 26 0000000000 65535 f +0000000016 00000 n +0000000144 00000 n +0000021707 00000 n +0000000000 00000 f +0000024726 00000 n +0000024540 00000 n +0000074969 00000 n +0000021758 00000 n +0000022120 00000 n +0000028374 00000 n +0000028261 00000 n +0000023430 00000 n +0000023979 00000 n +0000024027 00000 n +0000024610 00000 n +0000024641 00000 n +0000025124 00000 n +0000025502 00000 n +0000028448 00000 n +0000028656 00000 n +0000029633 00000 n +0000034894 00000 n +0000051182 00000 n +0000065251 00000 n +0000074992 00000 n +trailer <]>> startxref 75173 %%EOF \ No newline at end of file diff --git a/correctness-model/writeup/figures/rmw_atomicity.dot b/correctness-model/writeup/figures/rmw_atomicity.dot new file mode 100644 index 0000000..8b1547c --- /dev/null +++ b/correctness-model/writeup/figures/rmw_atomicity.dot @@ -0,0 +1,12 @@ +digraph rmw_atomicity { +margin=0; +ranksep=.1; + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="B: v.rmw()", shape=none]; +N3 [label="C: v.store(2)", shape=none]; + +N1 -> N2 [label="rf", color=red]; +N1 -> N3 [label="mo", color="#00aa00", weight=100]; +N2 -> N3 [color=invis]; +} diff --git a/correctness-model/writeup/figures/rmw_atomicity2.dot b/correctness-model/writeup/figures/rmw_atomicity2.dot new file mode 100644 index 0000000..483367c --- /dev/null +++ b/correctness-model/writeup/figures/rmw_atomicity2.dot @@ -0,0 +1,13 @@ +digraph rmw_atomicity2 { +margin=0; +ranksep=.1; + +//N1 [label="A: v.store(1)", shape=none, group="a"]; +N1 [style=invis, shape=none]; +N2 [label="B: v.rmw()", shape=none]; +N3 [label="C: v.store(2)", shape=none]; + +N1 -> N2 [color=invis]; +N1 -> N3 [color=invis, weight=100]; +N2 -> N3 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +} diff --git a/correctness-model/writeup/figures/rmw_mo.dot b/correctness-model/writeup/figures/rmw_mo.dot new file mode 100644 index 0000000..7a41997 --- /dev/null +++ b/correctness-model/writeup/figures/rmw_mo.dot @@ -0,0 +1,9 @@ +digraph rmw_mo { +margin=0 + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="B: v.rmw()", shape=none]; + +N1 -> N2 [label="rf", color=red]; +N1 -> N2 [color=invis, constraint=false]; +} diff --git a/correctness-model/writeup/figures/rmw_mo2.dot b/correctness-model/writeup/figures/rmw_mo2.dot new file mode 100644 index 0000000..0d4ce57 --- /dev/null +++ b/correctness-model/writeup/figures/rmw_mo2.dot @@ -0,0 +1,8 @@ +digraph rmw_mo2 { +margin=0 + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="B: v.rmw()", shape=none]; + +N1 -> N2 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +} diff --git a/correctness-model/writeup/figures/rr_mo.dot b/correctness-model/writeup/figures/rr_mo.dot new file mode 100644 index 0000000..179991e --- /dev/null +++ b/correctness-model/writeup/figures/rr_mo.dot @@ -0,0 +1,13 @@ +digraph rr_mo { +margin=0 + +N1 [label="X: v.store(1)", shape=none]; +N2 [label="Y: v.store(2)", shape=none]; +N3 [label="A: v.load()", shape=none]; +N4 [label="B: v.load()", shape=none]; + +N3 -> N4 [label="hb", color=blue]; +N1 -> N3 [label="rf", color=red, constraint=false]; +N2 -> N4 [label="rf", color=red, constraint=false]; +N1 -> N2 [color=invis]; +} diff --git a/correctness-model/writeup/figures/rr_mo2.dot b/correctness-model/writeup/figures/rr_mo2.dot new file mode 100644 index 0000000..fa0a6f7 --- /dev/null +++ b/correctness-model/writeup/figures/rr_mo2.dot @@ -0,0 +1,8 @@ +digraph rr_mo { +margin=0 + +N1 [label="X: v.store(1)", shape=none]; +N2 [label="Y: v.store(2)", shape=none]; + +N1 -> N2 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +} diff --git a/correctness-model/writeup/figures/rw_mo.dot b/correctness-model/writeup/figures/rw_mo.dot new file mode 100644 index 0000000..a8e99fe --- /dev/null +++ b/correctness-model/writeup/figures/rw_mo.dot @@ -0,0 +1,11 @@ +digraph rw_mo { +margin=0; +ranksep=.3; + +N1 [label="X: v.store(1)", shape=none]; +N2 [label="A: v.load()", shape=none]; +N3 [label="B: v.store(2)", shape=none]; + +N2 -> N3 [label="hb", color=blue]; +N1 -> N2 [label="rf", color=red, constraint=false]; +} diff --git a/correctness-model/writeup/figures/rw_mo2.dot b/correctness-model/writeup/figures/rw_mo2.dot new file mode 100644 index 0000000..e7f611a --- /dev/null +++ b/correctness-model/writeup/figures/rw_mo2.dot @@ -0,0 +1,12 @@ +digraph rw_mo2 { +margin=0; +ranksep=.3; + +N1 [label="X: v.store(1)", shape=none]; +N2 [label="A: v.load()", shape=none, fontcolor=invis]; +N3 [label="B: v.store(2)", shape=none]; + +N2 -> N3 [color=invis]; +N1 -> N2 [color=invis, constraint=false]; +N1 -> N3 [label="mo", color="#00aa00", style=dashed, penwidth=3, constraint=false]; +} diff --git a/correctness-model/writeup/figures/sc_mo.dot b/correctness-model/writeup/figures/sc_mo.dot new file mode 100644 index 0000000..5d5f6b1 --- /dev/null +++ b/correctness-model/writeup/figures/sc_mo.dot @@ -0,0 +1,9 @@ +digraph sc_mo { +margin=0 + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="B: v.store(2)", shape=none]; + +N1 -> N2 [label="sc", color=purple]; +N1 -> N2 [color=invis, constraint=false]; +} diff --git a/correctness-model/writeup/figures/sc_mo2.dot b/correctness-model/writeup/figures/sc_mo2.dot new file mode 100644 index 0000000..a7d37c3 --- /dev/null +++ b/correctness-model/writeup/figures/sc_mo2.dot @@ -0,0 +1,8 @@ +digraph sc_mo2 { +margin=0 + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="B: v.store(2)", shape=none]; + +N1 -> N2 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +} diff --git a/correctness-model/writeup/figures/sc_wr_mo.dot b/correctness-model/writeup/figures/sc_wr_mo.dot new file mode 100644 index 0000000..49a5f03 --- /dev/null +++ b/correctness-model/writeup/figures/sc_wr_mo.dot @@ -0,0 +1,13 @@ +digraph sc_wr_mo { +margin=0; +ranksep=.3; + +N1 [label="X: v.store(2)", shape=none]; +N2 [label="A: v.store(1, seq_cst)", shape=none]; +N3 [label="B: v.load(seq_cst)", shape=none]; + +N2 -> N3 [label="sc", color=purple]; +N1 -> N3 [label="rf", color=red]; +N2 -> N1 [color=invis, constraint=false]; +{rank=same; N1 N3} +} diff --git a/correctness-model/writeup/figures/sc_wr_mo2.dot b/correctness-model/writeup/figures/sc_wr_mo2.dot new file mode 100644 index 0000000..7efd810 --- /dev/null +++ b/correctness-model/writeup/figures/sc_wr_mo2.dot @@ -0,0 +1,13 @@ +digraph sc_wr_mo2 { +margin=0; +ranksep=.3; + +N1 [label="X: v.store(2)", shape=none]; +N2 [label="A: v.store(1, seq_cst)", shape=none]; +N3 [label="B: v.load(seq_cst)", shape=none, fontcolor=invis]; + +N2 -> N3 [color=invis]; +N1 -> N3 [color=invis]; +N2 -> N1 [label="mo", color="#00aa00", style=dashed, penwidth=3, constraint=false]; +{rank=same; N1 N3} +} diff --git a/correctness-model/writeup/figures/specdesign.pdf b/correctness-model/writeup/figures/specdesign.pdf new file mode 100644 index 0000000..203ab8d Binary files /dev/null and b/correctness-model/writeup/figures/specdesign.pdf differ diff --git a/correctness-model/writeup/figures/specworkflow.pdf b/correctness-model/writeup/figures/specworkflow.pdf new file mode 100644 index 0000000..72c4c78 Binary files /dev/null and b/correctness-model/writeup/figures/specworkflow.pdf differ diff --git a/correctness-model/writeup/figures/wf_sc_rf.dot b/correctness-model/writeup/figures/wf_sc_rf.dot new file mode 100644 index 0000000..95116d3 --- /dev/null +++ b/correctness-model/writeup/figures/wf_sc_rf.dot @@ -0,0 +1,16 @@ +digraph wf_sc_rf { +margin=0 +ranksep=.3; + +N1 [label="A: v.store(1, seq-cst)", shape=none]; +N2 [label="X: fence(seq-cst)", shape=none]; +N3 [label="B: v.store(2)", shape=none]; +N4 [label="Y: v.load()", shape=none]; + +//N1 -> N3 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +N1 -> N2 [label="sc", color=purple]; +N2 -> N4 [label="sb", color=blue, weight=100]; +N3 -> N4 [label="rf", color=red]; +N1 -> N3 [style=invis]; +{rank=same; N3 N4} +} diff --git a/correctness-model/writeup/figures/wf_sc_rf2.dot b/correctness-model/writeup/figures/wf_sc_rf2.dot new file mode 100644 index 0000000..1cdc907 --- /dev/null +++ b/correctness-model/writeup/figures/wf_sc_rf2.dot @@ -0,0 +1,16 @@ +digraph wf_sc_rf2 { +margin=0 +ranksep=.3; + +N1 [label="A: v.store(1, seq-cst)", shape=none]; +//N2 [label="X: fence(seq-cst)", shape=none]; +N3 [label="B: v.store(2)", shape=none]; +//N4 [label="Y: v.load()", shape=none]; + +N1 -> N3 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +//N1 -> N2 [label="sc", color=purple]; +//N2 -> N4 [label="sb", color=blue]; +//N3 -> N4 [label="rf", color=red, constraint=false]; +//N1 -> N3 [style=invis]; +//{rank=same; N1 N2} +} diff --git a/correctness-model/writeup/figures/workflow.pdf b/correctness-model/writeup/figures/workflow.pdf new file mode 100644 index 0000000..1c852cf Binary files /dev/null and b/correctness-model/writeup/figures/workflow.pdf differ diff --git a/correctness-model/writeup/figures/wr_mo.dot b/correctness-model/writeup/figures/wr_mo.dot new file mode 100644 index 0000000..d8267be --- /dev/null +++ b/correctness-model/writeup/figures/wr_mo.dot @@ -0,0 +1,13 @@ +digraph wr_mo { +margin=0; +ranksep=.3; + +N1 [label="X: v.store(2)", shape=none]; +N2 [label="A: v.store(1)", shape=none]; +N3 [label="B: v.load()", shape=none]; + +N2 -> N3 [label="hb", color=blue]; +N1 -> N3 [label="rf", color=red]; +N2 -> N1 [color=invis, constraint=false]; +{rank=same; N1 N3} +} diff --git a/correctness-model/writeup/figures/wr_mo2.dot b/correctness-model/writeup/figures/wr_mo2.dot new file mode 100644 index 0000000..182dea6 --- /dev/null +++ b/correctness-model/writeup/figures/wr_mo2.dot @@ -0,0 +1,13 @@ +digraph wr_mo2 { +margin=0; +ranksep=.3; + +N1 [label="X: v.store(2)", shape=none]; +N2 [label="A: v.store(1)", shape=none]; +N3 [label="B: v.load()", shape=none, fontcolor=invis]; + +N2 -> N3 [color=invis]; +N1 -> N3 [color=invis]; +N2 -> N1 [label="mo", color="#00aa00", style=dashed, penwidth=3, constraint=false]; +{rank=same; N1 N3} +} diff --git a/correctness-model/writeup/figures/ww_mo.dot b/correctness-model/writeup/figures/ww_mo.dot new file mode 100644 index 0000000..2e00cd9 --- /dev/null +++ b/correctness-model/writeup/figures/ww_mo.dot @@ -0,0 +1,9 @@ +digraph ww_mo { +margin=0 + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="B: v.store(2)", shape=none]; + +N1 -> N2 [label="hb", color=blue]; +N1 -> N2 [color=invis, constraint=false]; +} diff --git a/correctness-model/writeup/figures/ww_mo2.dot b/correctness-model/writeup/figures/ww_mo2.dot new file mode 100644 index 0000000..149c113 --- /dev/null +++ b/correctness-model/writeup/figures/ww_mo2.dot @@ -0,0 +1,8 @@ +digraph ww_mo2 { +margin=0 + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="B: v.store(2)", shape=none]; + +N1 -> N2 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +} diff --git a/correctness-model/writeup/figures/ww_sc_fence_first_collapse.dot b/correctness-model/writeup/figures/ww_sc_fence_first_collapse.dot new file mode 100644 index 0000000..c85988c --- /dev/null +++ b/correctness-model/writeup/figures/ww_sc_fence_first_collapse.dot @@ -0,0 +1,17 @@ +digraph ww_sc_fence_first_collapse { +margin=0 +ranksep=.3; + +N1 [label="A: v.store(1, seq-cst)", shape=none]; +//N2 [label="X: fence(seq-cst)", shape=none]; +N3 [label="Y: fence(seq-cst)", shape=none]; +N4 [label="B: v.store(2)", shape=none]; +N5 [style=invis, shape=point]; + +//N1 -> N4 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +N1 -> N3 [label="sc", color=purple]; +N3 -> N4 [label="sb", color=blue, weight=100]; +N1 -> N5 [style=invis]; +N5 -> N4 [style=invis]; +{rank=same; N4 N5} +} diff --git a/correctness-model/writeup/figures/ww_sc_fence_first_collapse2.dot b/correctness-model/writeup/figures/ww_sc_fence_first_collapse2.dot new file mode 100644 index 0000000..17d14a0 --- /dev/null +++ b/correctness-model/writeup/figures/ww_sc_fence_first_collapse2.dot @@ -0,0 +1,14 @@ +digraph ww_sc_fence_first_collapse2 { +margin=0 +//ranksep=.3; + +N1 [label="A: v.store(1, seq-cst)", shape=none]; +N2 [style=invis, shape=point]; +N3 [style=invis, shape=point]; +N4 [label="B: v.store(2)", shape=none]; + +N1 -> N4 [label="mo", color="#00aa00", style=dashed, penwidth=3, constraint=false]; +N1 -> N2 [color=invis, weight=100]; +N3 -> N4 [color=invis, weight=100]; +{rank=same; N2 N3} +} diff --git a/correctness-model/writeup/figures/ww_sc_fence_mo.dot b/correctness-model/writeup/figures/ww_sc_fence_mo.dot new file mode 100644 index 0000000..3edead9 --- /dev/null +++ b/correctness-model/writeup/figures/ww_sc_fence_mo.dot @@ -0,0 +1,18 @@ +digraph ww_sc_fence_mo { +margin=0 +ranksep=.3; + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="X: fence(seq-cst)", shape=none]; +N3 [label="Y: fence(seq-cst)", shape=none]; +N4 [label="B: v.store(2)", shape=none]; +N5 [style=invis, shape=point]; + +//N1 -> N4 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +N2 -> N3 [label="sc", color=purple]; +N1 -> N2 [label="sb", color=blue, weight=100]; +N3 -> N4 [label="sb", color=blue, weight=100]; +N2 -> N5 [style=invis]; +N5 -> N4 [style=invis]; +{rank=same; N4 N5} +} diff --git a/correctness-model/writeup/figures/ww_sc_fence_mo2.dot b/correctness-model/writeup/figures/ww_sc_fence_mo2.dot new file mode 100644 index 0000000..f1964fd --- /dev/null +++ b/correctness-model/writeup/figures/ww_sc_fence_mo2.dot @@ -0,0 +1,14 @@ +digraph ww_sc_fence_mo2 { +margin=0 +//ranksep=.3; + +N1 [label="A: v.store(1)", shape=none]; +N2 [style=invis, shape=point]; +N3 [style=invis, shape=point]; +N4 [label="B: v.store(2)", shape=none]; + +N1 -> N4 [label="mo", color="#00aa00", style=dashed, penwidth=3, constraint=false]; +N1 -> N2 [color=invis, weight=100]; +N3 -> N4 [color=invis, weight=100]; +{rank=same; N2 N3} +} diff --git a/correctness-model/writeup/figures/ww_sc_fence_second_collapse.dot b/correctness-model/writeup/figures/ww_sc_fence_second_collapse.dot new file mode 100644 index 0000000..6dde5c5 --- /dev/null +++ b/correctness-model/writeup/figures/ww_sc_fence_second_collapse.dot @@ -0,0 +1,17 @@ +digraph ww_sc_fence_second_collapse { +margin=0 +ranksep=.3; + +N1 [label="A: v.store(1)", shape=none]; +N2 [label="X: fence(seq-cst)", shape=none]; +//N3 [label="Y: fence(seq-cst)", shape=none]; +N4 [label="B: v.store(2, seq-cst)", shape=none]; +N5 [style=invis, shape=point]; + +//N1 -> N4 [label="mo", color="#00aa00", style=dashed, penwidth=3]; +N2 -> N4 [label="sc", color=purple]; +N1 -> N2 [label="sb", color=blue, weight=100]; +N2 -> N5 [style=invis]; +N5 -> N4 [style=invis]; +{rank=same; N4 N5} +} diff --git a/correctness-model/writeup/figures/ww_sc_fence_second_collapse2.dot b/correctness-model/writeup/figures/ww_sc_fence_second_collapse2.dot new file mode 100644 index 0000000..e7c0b35 --- /dev/null +++ b/correctness-model/writeup/figures/ww_sc_fence_second_collapse2.dot @@ -0,0 +1,14 @@ +digraph ww_sc_fence_second_collapse2 { +margin=0 +//ranksep=.3; + +N1 [label="A: v.store(1)", shape=none]; +N2 [style=invis, shape=point]; +N3 [style=invis, shape=point]; +N4 [label="B: v.store(2, seq-cst)", shape=none]; + +N1 -> N4 [label="mo", color="#00aa00", style=dashed, penwidth=3, constraint=false]; +N1 -> N2 [color=invis, weight=100]; +N3 -> N4 [color=invis, weight=100]; +{rank=same; N2 N3} +} diff --git a/correctness-model/writeup/formalization.tex b/correctness-model/writeup/formalization.tex new file mode 100644 index 0000000..aecb783 --- /dev/null +++ b/correctness-model/writeup/formalization.tex @@ -0,0 +1,92 @@ +\mysection{Formalization of Correctness Model}\label{sec:formalization} + +Unlike the SC memory model, applying linearizability can be complicated under +C/C++ by the fact that the C/C++ memory model allows atomic loads to read from +atomic stores that appear later in the trace and that it is in general +impossible to produce a sequential history that preserves program order for the +C/C++ memory model. Intuitively however, we can weaken some constraints, e.g. +the \textit{happens-before} edges in some cases, to generate a reordering of +ordering points such that they yield a sequential history that is consistent +with the specification. We formalize our approach as follow. + +We represent a trace as an \textit{execution graph}, where each node represents +each API method call with a set of ordering points, and edges between nodes are +derived from the \textit{happens-before} edges and the \textit{modification +order} edges between ordering points. We define \textit{opo} as the +\textit{ordering point order} relation between ordering point. Given two +operations $X$ and $Y$ that are both ordering points, the \textit{modification +order} edges are as follow: + +\squishcount + +\item {\bf Modification Order (write-write):} $X \xrightarrow{mo} Y \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (read-write):} $A \xrightarrow{mo} Y \ \wedge \ A \xrightarrow{rf} X \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (write-read):} $X \xrightarrow{mo} B \ \wedge \ B \xrightarrow{rf} Y \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (read-read):} $A \xrightarrow{mo} B \ \wedge \ A \xrightarrow{rf} X \ \wedge \ B \xrightarrow{rf} Y \Rightarrow X \xrightarrow{opo} Y$. +\countend + +\vspace{0.3cm} + +Intuitively, if method $A$ + +In order to relax the contraints on the original execution graph, we define an +action \textit{tranform} that can be performed on the graph as follow: + +\mypara{{\bf Hoisting loads:}} $ \forall X, Y, X \in +\textit{OrderingPoints}\xspace \wedge Y \in \textit{OrderingPoints}\xspace \wedge +address(X)\not=address(Y) \wedge +X \relation{hb} Y \wedge Y \in \textit{LoadOps}\xspace +\Rightarrow \forall Z, Z +$. + + +\begin{figure}[!htbp] +\begin{algorithmic}[1] +\Function{InferParams}{testcases, initialParams} +\State inputParams := initialParams +\If{inputParams is empty} +\State inputParams := the weakest parameters \label{line:startWithRelaxed} +\EndIf +\ForAll{test case $t$ in testcases} +\State candidates := inputParams \label{line:setOfCandidates} +\State results := \{\} +\While{candidates is not empty} +\State Candidate $c$ := pop from candidates +\State run \cdschecker with $c$ and check SC \label{line:findSCViolation} +\If{$\exists$ SC violation $v$} +\State \Call {StrengthenParam}{$v$, $c$, candidates} \label{line:callStrengthenParam} +\Else +\State results += \Call {WeakenOrderParams}{$c$} \label{line:weakenOrderParams} +\EndIf +\EndWhile +\State inputParams := results \label{line:tmpOutputAsInput} +\EndFor +\State \Return{results} \label{line:finalResults} +\EndFunction + +\Procedure{StrengthenParam}{v, c, candidates} +\While{$\exists$ a fix $f$ for violation $v$} +\State possible\_repairs := strengthen $c$ with fix $f$ \label{line:strengthenParam} +\State candidates += possible\_repairs \label{line:possibleFixes} +\EndWhile +\EndProcedure + +\end{algorithmic} +\caption{\label{fig:algorithmfence}Algorithm for inferring order parameters} +\end{figure} + + + +\mypara{\bf Generating the Reordering:} +The \TOOL checker first builds an execution graph where the nodes are +method calls and the edges represent the +$\reltext{opo}$ ordering of the ordering points of the methods that +correspond to the source and destination nodes. Assuming the absence +of cycles in the execution graph, +the $\reltext{opo}$ ordering is used to generate the sequential history. +The \TOOL checker +topologically sorts the graph to generate the equivalent sequential +execution. diff --git a/correctness-model/writeup/implementation.tex b/correctness-model/writeup/implementation.tex new file mode 100644 index 0000000..bac8f73 --- /dev/null +++ b/correctness-model/writeup/implementation.tex @@ -0,0 +1,151 @@ +\mysection{Implementation}\label{sec:implementation} + +The goal of the \TOOL specification language is to +enable developers to write specifications against which concurrent data structures can be tested. +We can ensure a concurrent data structure is correct with +respect to an equivalent sequential data structure if for each +execution of the concurrent data structure, the equivalent sequential +history for the equivalent sequential data structure yields the same +results. + +The execution space for many concurrent data +structures is unbounded, meaning that in practice we cannot +verify correctness by checking individual executions. However, the +specifications can be used for unit testing. In practice, +many bugs can be exposed by model checking unit tests for +concurrent data structures. We have implemented the \TOOL checker as a +unit testing tool built upon the \cdschecker framework. \TOOL +can exhaustively explore all behaviors for unit tests and provide +developers with diagnostic reports for executions that violate their +specification. + + +\mysubsection{Model Checker Framework} + +The \TOOL checker takes as input a complete execution from the +\cdschecker model checker. The \cdschecker framework operates at the abstraction level of +individual atomic operations and thus has neither information about +method calls nor which atomic operations serve as ordering points. +Thus, we extend the framework by adding {\it annotation} operations to +\cdschecker's traces, which record the necessary information to check the specifications +but have no effect on other operations. The \TOOL compiler +inserts code to generate the annotation actions to communicate to the \TOOL checker +plugin the critical events for checking the \TOOL specification. +These annotation actions then appear in \cdschecker's list of atomic +operations and make it convenient for \TOOL to construct +a sequential history from the execution because for any given method +call, its invocation event, its ordering points, and its response event are +sequentially ordered in the list. + +\mysubsection{Specification Compiler} + +The specification compiler translates an annotated +C/C++ program into an instrumented C/C++ program that will generate +execution traces containing the dynamic information needed to construct the sequential history +and check the specification assertions. We next +describe the type of annotation actions that the \TOOL compiler +inserts into the instrumented program. + +\mypara{\bf Ordering Points:} Ordering points have a conditional guard expression +and a label. Potential ordering point annotation actions are +inserted immediately after the atomic operation that serves as +the potential ordering point. Ordering point check annotation actions are +inserted where they appear. + +\mypara{\bf Method Boundary:} To identify a method's boundaries, \TOOL +inserts {\it method\_begin} and {\it method\_end} annotations at the +beginning and end of methods. + +\mypara{\bf Sequential States and Methods:} Since checking occurs +after \cdschecker has completed an execution, the annotation +actions stores the values of any variables in the concurrent data +structure that the annotations reference. + +\mypara{\bf Side Effects and Assertions:} Side effects and assertions perform +their checks after an execution. The side effects and assertions are compiled +into methods and the equivalent sequential data structure's states are accessible to these methods. +With this encapsulation, +the \TOOL checker simply calls these functions to implement the side effects and assertions. + +\mypara{\bf Synchronization Checks:} The \TOOL checker performs +synchronization checks in two parts: compiling the rules and runtime data +collection. First, the \TOOL compiler numbers all methods and happens-before +checks uniquely. For example, the rule ``\code{Update->Read}'' can be represented as (1, 0, 2, 0), +which means instances of method 1 that satisfy condition 0 should {\it + synchronize with} instances method 2 that satisfy condition 0. In this case, + condition 0 means \code{true}. Then, +the \TOOL compiler generates code that communicates the +synchronization rules by passing an array of integer pairs. Runtime +collection is then implemented by performing the condition check at +each method invocation or response and then passing the method number +and happens before condition if the check is satisfied. + +\mysubsection{Dynamic Checking} + +At this point, we have an execution trace with the necessary +annotations to construct a sequential history and to check +the execution's correctness. However, before constructing the +sequential history, the \TOOL plugin first collects +the necessary information for each method call, which is the {\it + method\_begin} annotation, the ordering point annotations, the +happens-before checks, and the {\it method\_end} annotations. Since +all of the operations in the trace have thread +identifiers it is straightforward to extract the operations between the {\it + method\_begin} and {\it method\_end} annotations. + +\mypara{\bf Reorder Method Calls:} As discussed above, determining the +ordering of the ordering points is non-trivial under the C/C++ memory +model. This can be complicated by the fact that the C/C++ memory model +allows atomic loads to read from atomic stores that appear later in +the trace and that it is in general impossible to produce a sequential +history that preserves program order for the C/C++ memory model. + +However, we can still leverage the reads-from relation and the +modification-order relation to order the ordering points that appear +in typical data structures. \TOOL uses the following rules to +generate an ordering-point ordering $\reltext{opo}$ relation on +ordering points. Given two operations $X$ and $Y$ that are both +ordering points: + +\squishcount +\item {\bf Reads-From:} $X \xrightarrow{rf} Y \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (write-write):} $X \xrightarrow{mo} Y \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (read-write):} $A \xrightarrow{mo} Y \ \wedge \ A \xrightarrow{rf} X \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (write-read):} $X \xrightarrow{mo} B \ \wedge \ B \xrightarrow{rf} Y \Rightarrow X \xrightarrow{opo} Y$. + +\item {\bf Modification Order (read-read):} $A \xrightarrow{mo} B \ \wedge \ A \xrightarrow{rf} X \ \wedge \ B \xrightarrow{rf} Y \Rightarrow X \xrightarrow{opo} Y$. +\countend + +\mypara{\bf Generating the Reordering:} +The \TOOL checker first builds an execution graph where the nodes are +method calls and the edges represent the +$\reltext{opo}$ ordering of the ordering points of the methods that +correspond to the source and destination nodes. Assuming the absence +of cycles in the execution graph, +the $\reltext{opo}$ ordering is used to generate the sequential history. +The \TOOL checker +topologically sorts the graph to generate the equivalent sequential +execution. + +When \TOOL fails to order two ordering points, the operations often commute. Thus, +if multiple histories satisfy the constraints of \reltext{opo}, by default we generally randomly select one. However, when those operations +do not commute, we provide developers with different options: (1) they can +add additional ordering points to order the two nodes or +(2) they can run \TOOL in either of the following modes: (a) \textit{loosely +exhaustive} mode --- \TOOL explores +all possible histories and only requires that there exists some history that passes the checks or +(b) \textit{strictly exhaustive} mode --- \TOOL explores all possible histories and requires all histories pass the checks. + + +\mypara{\bf Synchronization Checks:} Synchronization properties are +specified using the IDs and conditions of method calls, and we have that information +available after \TOOL constructs the sequential history and checks the preconditions +and postconditions. For two specific method calls $c_1$ and $c_2$, we can ensure +$c_1$ synchronizes with $c_2$ by ensuring the annotation \code{c1\_begin} +happens-before the annotation \code{c2\_end} because any operations +sequenced-before \code{c1\_begin} should happen-before any operations +sequenced-after \code{c2\_end} according to the C/C++11 memory model. + diff --git a/correctness-model/writeup/introduction.tex b/correctness-model/writeup/introduction.tex new file mode 100644 index 0000000..e0c9c70 --- /dev/null +++ b/correctness-model/writeup/introduction.tex @@ -0,0 +1,226 @@ +\mysection{Introduction}\label{sec:introduction} + +Concurrent data structure design can improve scalability by supporting +multiple simultaneous operations, reducing memory coherence traffic, and reducing the time taken by an +individual data structure operation. Researchers have developed many concurrent +data structure designs with these goals~\cite{rcu, +lockfreequeue}. Concurrent data structures often use sophisticated techniques +including low-level atomic instructions (e.g., compare and +swap), careful reasoning about the order of loads and +stores, and fine-grained locking. For example, while the standard Java hash +table implementation can limit scalability to a handful of cores, +more sophisticated concurrent hash tables can scale to many hundreds of +cores~\cite{javaConcurrentHashMap}. + +The C/C++ standard committee extended the C and C++ languages with +support for low-level atomic operations in the C/C++11 +standard~\cite{cpp11spec, c11spec, boehmpldi} to enable developers to +write portable implementations of concurrent data structures. To support the relaxations typically performed by compilers and processors, +the +C/C++ memory model provides weaker semantics than sequential +consistency~\cite{scmemorymodel} and as a result, correctly using +these operations is challenging. Developers must not only +reason about potential interleavings, but also about how the processor +and compiler might reorder memory operations. Even experts make +subtle errors when reasoning about such memory models. + +Researchers have developed tools for exploring the behavior of code +under the C/C++ memory model including \cdschecker~\cite{cdschecker}, {\sc +Cppmem}~\cite{c11popl}, and Relacy~\cite{relacy}. These tools explore +behaviors that are allowed under the C/C++ memory model. While these +tools can certainly be useful for exploring executions, they can be +challenging to use for testing as they don't provide support (other +than assertions) for specifying the behavior of data structures. +Using assertions can be challenging as different interleavings or +reorderings legitimately produce different behaviors, and it can be +very difficult to code assertions to check the output of a test case for an +arbitrary (unknown) execution. + +This paper presents \TOOL, a specification language and specification +checking tool that is designed to be used in conjunction with model +checking tools. We have implemented it as a plugin for +the \cdschecker model checker. + +\mysubsection{Background on Specifying the Correctness of Concurrent Data Structures} + +Researchers have developed several techniques for specifying +correctness properties of concurrent data structures written for strong memory +models. While these techniques cannot handle the behaviors typically +exhibited by relaxed data structure implementations, they provide insight into +intuitive approaches to specifying concurrent data structure behavior. + +One approach for specifying the correctness of concurrent data +structures is in terms of equivalent sequential executions of either +the concurrent data structure or a simplified sequential version. The +problem then becomes how do we map a concurrent execution to an equivalent +sequential execution? A common criterion is {\it + linearizability} --- linearizability simply states that a concurrent +operation can be viewed as taking effect at some time between its +invocation and its return (or response)~\cite{linearizableref}. + +An {\it equivalent sequential data structure} is a sequential version +of a concurrent data structure that can be used to express correctness +properties by relating executions of the original concurrent data +structure with executions of the equivalent sequential data structure. +The equivalent sequential data structure is often simpler, and in +many cases one can simply use existing well-tested implementations from the +STL library. + +An execution {\it history} is a total order of method invocations and +responses. A {\it sequential history} is one where all invocations are +followed by the corresponding responses immediately. A concurrent +execution is correct if its behavior is consistent with its equivalent +sequential history replayed on the equivalent sequential data +structure. A concurrent object is linearizable if for all executions: +\squishcount +\item Each method call appears to take effect instantaneously at some + point between its invocation and response. + +\item The invocations and responses can be reordered to yield a + sequential history under the rule that an invocation cannot be + reordered before the preceding responses. + +\item The concurrent execution yields the same behavior as + the sequential history. +\countend + +A weaker variation of linearization is {\it sequential +consistency}\footnote{It is important to note that the term sequential +consistency in the literature is applied to both the consistency model +that data structures expose clients to as well as the guarantees that +the underlying memory system provides for load and store operations.}. +Sequential consistency only requires that there exists a sequential +history that is consistent with the {\it program order} (the intra-thread +order). This ordering does not need to be consistent with the order +that the operations were actually issued in. + +Line-Up~\cite{lineup}, Paraglider~\cite{VechevMCLinear}, and +VYRD~\cite{vyrd} leverage linearizability to test concurrent data +structures. {\bf Unfortunately, efficient implementations of many +common data structures, e.g., RCU~\cite{rcu}, MS +Queue~\cite{lockfreequeue}, etc., for the C/C++ memory model are +neither linearizable nor sequentially consistent! Thus previous tools +cannot check such data structures under the C/C++ memory model.} + +\mysubsection{New Challenges from the C/C++ Memory Model} +\label{sec:introNewChallenges} + +The C/C++ memory model brings the following two key challenges that prevent the application of previous approaches to specifying the concurrent data structures to this setting: + +\squishcount +\item {\bf Relaxed Executions Break Existing Data Structure Consistency Models:} +C/C++ data structures often expose clients to weaker (non-SC) +behaviors to gain performance. A common guarantee is to provide +happens-before synchronization between operations that implement +updates and the operations that read those updates. These data +structures often do not guarantee that different threads observe updates in the +same order --- in other words the data structures may expose clients +to weaker consistency models than sequential consistency. For +example, even when one uses the relatively strong \code{acquire} +and \code{release} memory orderings in C++, it is possible for two +different threads to observe two stores happening in +different orders, i.e., executions can fail the IRIW test. +Thus many data structures legitimately admit executions for which there +are no sequential histories that preserve program order. + +Like many other relaxed memory models, the C/C++ memory model does +not include a total order over all memory operations, thus even +further complicating the application of traditional approaches to correctness, +e.g., linearization cannot be applied. In +particular the approaches that relate the behaviors of concurrent data +structures to analogous sequential data structures break down due to +the absence of a total ordering of the memory operations. While many +of the dynamic tools~\cite{cdschecker,relacy} for exploring the +behavior of code under relaxed models do as a practical matter +print out an execution in some order, this order is to some degree arbitrary as relaxed memory models +generally make it possible for a data structure operation to see the +effects of operations that appear later in any such an order (e.g., a load +can read from a store that appears later in the order). Instead of a total order, the C/C++ +memory model is formulated as a graph of memory operations +with several partial orders defined in this graph. + +\item {\bf Constraining Reorderings (Specifying Synchronization Properties):} +Synchronization\footnote{Synchronization here is not mutual exclusion, + but rather a lower-level property that captures which stores must be + visible to a thread. In other words, it constrains which + reorderings can be performed by a processor or compiler.} in C/C++ + provides an ordering between memory operations to different + locations. Concurrent data structures must establish + synchronization or they potentially expose their users to highly + non-intuitive behavior that is likely to break client code. For + example, consider the case of a concurrent queue that does not + establish synchronization between enqueue and dequeue operations. Consider the + following sequence of operations: (1) thread A initializes the fields of a new + object X; (2) thread A enqueues the reference to X in such a queue + (3) thread B dequeues the reference to X; (4) thread B reads the + fields of X through the dequeued reference. In (4), thread B could fail to see the + initializing writes from (1). + This surprising behavior could occur if the compiler or CPU could + reorder the initializing writes to be executed after the enqueue + operation. If the fields are non-atomic, such loads are considered + data races and violate the data race free requirement of the C/C++ + language standard and thus the program has no semantics. + +The C/C++ memory model formalizes synchronization in terms of a {\it + happens-before} relation. The C/C++ happens-before relationship +is a partial order over memory accesses. If memory access {\it x} +happens before memory access {\it y}, it means that the effects of +{\it x} must be ordered before the effects of {\it y}. +\countend + +\begin{figure} + \centering +\vspace{-.3cm} + \includegraphics[scale=0.35]{figures/specworkflow} +\vspace{-.4cm} + \caption{\label{fig:specworkflow} \TOOL system overview} +\vspace{-.4cm} +\end{figure} + +\mysubsection{Specification Language and Tool Support} + +Figure~\ref{fig:specworkflow} presents an overview of the \TOOL system. After +implementing a concurrent data structure, developers annotate their code +with a \TOOL specification. To test their implementation, +developers compile the data structure with the \TOOL specification +compiler to extract the specification and generate a program that is instrumented with specification checks. + Then, developers compile the instrumented program with a +standard C/C++ compiler. Finally, developers run the binary under +the \TOOL checker. \TOOL +then exhaustively explores the behaviors of the specific unit test and +generates diagnostic reports for any executions that violate the specification. + +\mysubsection{Contributions} + +This paper makes the following contributions: + +\squishlist +\item {\bf Specification Language:} It introduces a specification + language that enables developers to write specifications of + concurrent data structures developed for a relaxed memory model in a + simple fashion that capture the key correctness properties. Our + specification language is the first to our knowledge that supports concurrent data + structures that use C/C++ atomic operations. + +\item {\bf A Technique to Relate Concurrent Executions to Sequential Executions:} It presents an approach to order the memory operations for the + C/C++ model, which lacks a definition of a trace and for which one + generally cannot even construct a total order of atomic operations + that is consistent with the program order. The generated + sequential execution by necessity does not always maintain program order. + +\item {\bf Synchronization Properties:} It presents (a) constructs + for specifying the happens before relations that a data structure + should establish, and (b) tool support for checking these properties and + exposing synchronization related bugs. + +\item {\bf Tool for Checking C/C++ Data Structures Against Specifications:} +\TOOL is the first tool to our knowledge that can check concurrent data +structures that exhibit relaxed behaviors against specifications that are +specified in terms of intuitive sequential executions. + +\item {\bf Evaluation:} It shows that the \TOOL specification language can express key correctness properties for a +set of real-world concurrent data structures, that our tool can detect + bugs, and that our tool can unit test real world data structures + with reasonable performance. +\squishend + diff --git a/correctness-model/writeup/issta15_submission.pdf b/correctness-model/writeup/issta15_submission.pdf new file mode 100644 index 0000000..84e6eed Binary files /dev/null and b/correctness-model/writeup/issta15_submission.pdf differ diff --git a/correctness-model/writeup/makefile b/correctness-model/writeup/makefile new file mode 100644 index 0000000..6c222e4 --- /dev/null +++ b/correctness-model/writeup/makefile @@ -0,0 +1,94 @@ +LATEX := pdflatex -halt-on-error + +FIGURES = + +#FIGHEAD = #figures/figHeader.dotpiece +#FIGFOOT = #figures/figFooter.dotpiece + + +.SUFFIXES : .tex .bbl .aux .eps .ps .dot .pdf + +.tex.bbl: + bibtex $* + +.tex.aux: + latex $* + +%.eps: %.ps + ps2epsi $< $@ + +%.ps: %.dot + dot -Tps $< -o $@ + +%.pdf: %.dot + dot -Tpdf $< -o $@ + +%.png: %.dot + dot -Tpng $< -o $@ + +%.aux: %.tex + $(LATEX) $< + +#%.dot: %.dotpiece $(FIGHEAD) $(FIGFOOT) +# cat $(FIGHEAD) $< $(FIGFOOT) > $@ + + + +# latex the paper (default) +new: biblinks newpaper + +# latex + bibtex the paper +bib: bibpaper newpaper + +# save a backup of tex files in BACKUPS +bak: backuppaper + + +biblinks: figures + $(LATEX) paper.tex + bibtex paper.aux + $(LATEX) paper.tex + + +newpaper: figures biblinks + $(LATEX) paper.tex + + +bibpaper: paper.aux paper.bbl + latex paper.tex + + +backuppaper: + @if [ ! -d BACKUPS ]; then mkdir BACKUPS; fi; \ + crtdate=`date '+%m-%d-%y--%H:%M:%S'`; \ + bakdir=BACKUPS/$$crtdate; \ + mkdir $$bakdir; \ + cp *.tex $$bakdir + +spell: + @for i in *.tex; \ + do if [ $$i != "paper.tex" ]; then aspell -c $$i -p ./spell.lst; fi;\ + done + + +total: + latex paper.tex + bibtex paper.aux + latex paper.tex + latex paper.tex + pdflatex paper.tex + + +figures: $(FIGURES) + + +errinjfig: + gnuplot err-inj-fig/nve2latex.cmds + epstopdf injectErrorFig.eps + mv injectErrorFig.* figures + + +clean: + rm -f *.dvi *.log *.aux *.blg *.bbl *~ err-inj-fig/*~ + rm -f paper.ps paper.pdf + rm -f injectErrorFig* diff --git a/correctness-model/writeup/memorymodel.tex b/correctness-model/writeup/memorymodel.tex new file mode 100644 index 0000000..73e2e8e --- /dev/null +++ b/correctness-model/writeup/memorymodel.tex @@ -0,0 +1,102 @@ +\section{C/C++ Memory Model}\label{sec:memorymodel} + +We next briefly summarize the key aspects of the C/C++ memory model. The +memory model describes a set of atomic operations and the +corresponding allowed behaviors of programs that utilize them. A more +detailed formal treatment of the memory model~\cite{c11popl} and a +more detailed informal description~\cite{cpp11spec,c11spec} are +available in the literature. Any operation on an atomic object will +have one of six \textit{memory orders}, each of which falls into one +or more of the following categories. + +\begin{description} + + \item[relaxed:] + \code{\small{memory\_order\_relaxed}} -- + weakest ordering + + \item[release:] + \code{\small{memory\_order\_release}}, + \code{\small{memory\_order\_acq\_rel}}, and + \code{\small{memory\_order\_seq\_cst}} -- + a store-release may form release/consume or release/acquire synchronization + + \item[consume:\footnotemark{}] + \code{\small{memory\_order\_consume}} -- + a load-consume may form release/consume synchronization + + \footnotetext{Consume is not broadly supported by compilers due to + challenges associate with preserving data dependencies and is + unlikely to provide significant performance gains on x86 + hardware. We take the same approach as many compilers and treat + consumes as acquires.} + + \item[acquire:] + \code{\small{memory\_order\_acquire}}, + \code{\small{memory\_order\_acq\_rel}}, and + \code{\small{memory\_order\_seq\_cst}} -- + a load-acquire may form release/acquire synchronization + + \item[seq-cst:] + \code{\small{memory\_order\_seq\_cst}} -- + strongest ordering + +\end{description} + +\subsection{Relations}\label{sec:model-relations} + +The C/C++ memory model expresses program behavior in the form +of binary relations or orderings. The following subsections will briefly +summarize the relevant relations. Much of this discussion resembles the preferred model from the +formalization in \cite{c11popl}. + +\mypara{\bf Sequenced-Before:} +The order of program operations within a single thread of execution establishes an intra-thread +\textit{sequenced-before} (\reltext{sb}) relation. +Note that while some operations in C/C++ provide no +intra-thread ordering---the equality operator (\code{==}), for example---we +ignore this detail and assume that \reltext{sb} totally orders all operations in +a thread. + +\mypara{\bf Reads-From:} +The \textit{reads-from} (\reltext{rf}) relation consists of store-load pairs $(X, Y)$ +such that $Y$ reads its value from the effect of $X$---or $X \relation{rf} Y$. In the +C/C++ memory model, this relation is non-trivial, as a given load operation may +read from one of many potential stores in the program execution. + +\mypara{\bf Synchronizes-With:} +The \textit{synchronizes-with} (\reltext{sw}) relation captures +synchronization that occurs when certain atomic operations interact across two +threads. For instance, release/acquire synchronization occurs between a pair of +atomic operations on the same object: a store-release $X$ and a load-acquire +$Y$. If $Y$ reads from $X$, then $X$ synchronizes with $Y$---or $X +\relation{sw} Y$. + +\mypara{\bf Happens-Before:} +In \TOOL, we avoid consume operations, and so the \textit{happens-before} +(\reltext{hb}) relation is simply the transitive closure of \reltext{sb} and +\reltext{sw}. + +The \reltext{hb} relation restricts the stores that loads can read +from. For example, if we have two stores $X$ and $Y$ and +a load $Z$ to the same memory location and $X \relation{hb} Y +\relation{hb} Z$, then $Z$ cannot read from $X$. + +\mypara{\bf Sequential Consistency:} +All seq-cst operations in a program execution form a +total ordering (\reltext{sc}) so that, for instance, a seq-cst load may not read +from a seq-cst store prior to the most recent store (to the same location) in +the \reltext{sc} ordering, nor from any store that happens before that store. +The \reltext{sc} order must be consistent with \reltext{hb}. + +\mypara{\bf Modification Order:} +Each atomic object in a program execution has an associated \textit{modification order} +(\reltext{mo})---a total order of all stores to that object---which +informally represents a memory-coherent ordering in which those stores may be observed by +the rest of the program. +In general the modification orders for all objects cannot be combined +to form a consistent total ordering. + + + + diff --git a/correctness-model/writeup/paper.aux b/correctness-model/writeup/paper.aux new file mode 100644 index 0000000..605a7ee --- /dev/null +++ b/correctness-model/writeup/paper.aux @@ -0,0 +1,75 @@ +\relax +\providecommand\hyper@newdestlabel[2]{} +\providecommand\HyperFirstAtBeginDocument{\AtBeginDocument} +\HyperFirstAtBeginDocument{\ifx\hyper@anchor\@undefined +\global\let\oldcontentsline\contentsline +\gdef\contentsline#1#2#3#4{\oldcontentsline{#1}{#2}{#3}} +\global\let\oldnewlabel\newlabel +\gdef\newlabel#1#2{\newlabelxx{#1}#2} +\gdef\newlabelxx#1#2#3#4#5#6{\oldnewlabel{#1}{{#2}{#3}}} +\AtEndDocument{\ifx\hyper@anchor\@undefined +\let\contentsline\oldcontentsline +\let\newlabel\oldnewlabel +\fi} +\fi} +\global\let\hyper@last\relax +\gdef\HyperFirstAtBeginDocument#1{#1} +\providecommand\HyField@AuxAddToFields[1]{} +\providecommand\HyField@AuxAddToCoFields[2]{} +\citation{rcu} +\citation{lockfreequeue} +\citation{javaConcurrentHashMap} +\citation{cpp11spec} +\citation{c11spec} +\citation{boehmpldi} +\citation{scmemorymodel} +\citation{cdschecker} +\citation{c11popl} +\citation{relacy} +\citation{linearizableref} +\@writefile{toc}{\contentsline {section}{\numberline {1}Introduction}{\thepage }{section.1}} +\newlabel{sec:introduction}{{1}{\thepage }{Introduction}{section.1}{}} +\@writefile{toc}{\contentsline {subsection}{\numberline {1.1}Background on Specifying the Correctness of Concurrent Data Structures}{\thepage }{subsection.1.1}} +\citation{lineup} +\citation{VechevMCLinear} +\citation{vyrd} +\citation{rcu} +\citation{lockfreequeue} +\citation{cdschecker} +\citation{relacy} +\@writefile{toc}{\contentsline {subsection}{\numberline {1.2}New Challenges from the C/C++ Memory Model}{\thepage }{subsection.1.2}} +\newlabel{sec:introNewChallenges}{{1.2}{\thepage }{New Challenges from the C/C++ Memory Model}{subsection.1.2}{}} +\@writefile{lof}{\contentsline {figure}{\numberline {1}{\ignorespaces \textsc {CDSSpec}\xspace system overview}}{\thepage }{figure.1}} +\newlabel{fig:specworkflow}{{1}{\thepage }{\TOOL system overview}{figure.1}{}} +\@writefile{toc}{\contentsline {subsection}{\numberline {1.3}Specification Language and Tool Support}{\thepage }{subsection.1.3}} +\@writefile{toc}{\contentsline {subsection}{\numberline {1.4}Contributions}{\thepage }{subsection.1.4}} +\bibstyle{abbrv} +\bibdata{confstrs-long,paper} +\bibcite{cpp11spec}{1} +\bibcite{c11spec}{2} +\bibcite{c11popl}{3} +\bibcite{boehmpldi}{4} +\bibcite{lineup}{5} +\bibcite{rcu}{6} +\bibcite{vyrd}{7} +\bibcite{linearizableref}{8} +\bibcite{scmemorymodel}{9} +\bibcite{javaConcurrentHashMap}{10} +\@writefile{toc}{\contentsline {section}{\numberline {2}Formalization of Correctness Model}{\thepage }{section.2}} +\newlabel{sec:formalization}{{2}{\thepage }{Formalization of Correctness Model}{section.2}{}} +\newlabel{line:startWithRelaxed}{{4}{\thepage }{Formalization of Correctness Model}{Item.9}{}} +\newlabel{line:setOfCandidates}{{7}{\thepage }{Formalization of Correctness Model}{Item.9}{}} +\newlabel{line:findSCViolation}{{11}{\thepage }{Formalization of Correctness Model}{Item.9}{}} +\newlabel{line:callStrengthenParam}{{13}{\thepage }{Formalization of Correctness Model}{Item.9}{}} +\newlabel{line:weakenOrderParams}{{15}{\thepage }{Formalization of Correctness Model}{Item.9}{}} +\newlabel{line:tmpOutputAsInput}{{18}{\thepage }{Formalization of Correctness Model}{Item.9}{}} +\newlabel{line:finalResults}{{20}{\thepage }{Formalization of Correctness Model}{Item.9}{}} +\newlabel{line:strengthenParam}{{24}{\thepage }{Formalization of Correctness Model}{Item.9}{}} +\newlabel{line:possibleFixes}{{25}{\thepage }{Formalization of Correctness Model}{Item.9}{}} +\@writefile{lof}{\contentsline {figure}{\numberline {2}{\ignorespaces Algorithm for inferring order parameters}}{\thepage }{figure.2}} +\newlabel{fig:algorithmfence}{{2}{\thepage }{Algorithm for inferring order parameters}{figure.2}{}} +\@writefile{toc}{\contentsline {section}{\numberline {3}References}{\thepage }{section.3}} +\bibcite{lockfreequeue}{11} +\bibcite{cdschecker}{12} +\bibcite{VechevMCLinear}{13} +\bibcite{relacy}{14} diff --git a/correctness-model/writeup/paper.bbl b/correctness-model/writeup/paper.bbl new file mode 100644 index 0000000..e7f99db --- /dev/null +++ b/correctness-model/writeup/paper.bbl @@ -0,0 +1,84 @@ +\begin{thebibliography}{10} + +\bibitem{cpp11spec} +{ISO/IEC 14882:2011}, {Information} technology -- programming languages -- + {C++}. + +\bibitem{c11spec} +{ISO/IEC 9899:2011}, {Information} technology -- programming languages -- {C}. + +\bibitem{c11popl} +M.~Batty, S.~Owens, S.~Sarkar, P.~Sewell, and T.~Weber. +\newblock Mathematizing {C++} concurrency. +\newblock In {\em Proceedings of the Symposium on Principles of Programming + Languages}, 2011. + +\bibitem{boehmpldi} +H.~J. Boehm and S.~V. Adve. +\newblock Foundations of the {C++} concurrency memory model. +\newblock In {\em Proceedings of the 2008 ACM SIGPLAN Conference on Programming + Language Design and Implementation}, 2008. + +\bibitem{lineup} +S.~Burckhardt, C.~Dern, M.~Musuvathi, and R.~Tan. +\newblock Line-up: A complete and automatic linearizability checker. +\newblock In {\em Proceedings of the 2010 ACM SIGPLAN Conference on Programming + Language Design and Implementation}, 2010. + +\bibitem{rcu} +M.~Desnoyers, P.~E. McKenney, A.~S. Stern, M.~R. Dagenais, and J.~Walpole. +\newblock User-level implementations of read-copy update. +\newblock {\em IEEE Transactions on Parallel and Distributed Systems}, 2011. + +\bibitem{vyrd} +T.~Elmas, S.~Tasiran, and S.~Qadeer. +\newblock {VYRD}: Verifying concurrent programs by runtime refinement-violation + detection. +\newblock In {\em Proceedings of the 2005 ACM SIGPLAN Conference on Programming + Language Design and Implementation}, 2005. + +\bibitem{linearizableref} +M.~Herlihy and J.~Wing. +\newblock Linearizability: a correctness condition for concurrent objects. +\newblock {\em ACM Transactions on Programming Languages and Systems}, + 12(3):463--492, July 1990. + +\bibitem{scmemorymodel} +L.~Lamport. +\newblock How to make a multiprocessor computer that correctly executes + multiprocess programs. +\newblock {\em IEEE Transactions on Computers}, 28(9):690--691, Sept. 1979. + +\bibitem{javaConcurrentHashMap} +D.~Lea. +\newblock util.concurrent.{ConcurrentHashMap} in java.util.concurrent the {Java + Concurrency Package}. +\newblock + \url{http://docs.oracle.com/javase/8/docs/api/java/util/concurrent/ConcurrentHashMap.html}. + +\bibitem{lockfreequeue} +M.~M. Michael and M.~L. Scott. +\newblock Simple, fast, and practical non-blocking and blocking concurrent + queue algorithms. +\newblock In {\em Proceedings of the Fifteenth Annual ACM Symposium on + Principles of Distributed Computing}, 1996. + +\bibitem{cdschecker} +B.~Norris and B.~Demsky. +\newblock {CDSChecker}: Checking concurrent data structures written with + {C/C++} atomics. +\newblock In {\em Proceeding of the 28th ACM SIGPLAN Conference on + Object-Oriented Programming, Systems, Languages, and Applications}, 2013. + +\bibitem{VechevMCLinear} +M.~Vechev, E.~Yahav, and G.~Yorsh. +\newblock Experience with model checking linearizability. +\newblock In {\em International SPIN Workshop on Model Checking Software}, + 2009. + +\bibitem{relacy} +D.~Vyukov. +\newblock Relacy race detector. +\newblock \url{http://relacy.sourceforge.net/}, 2011 Oct. + +\end{thebibliography} diff --git a/correctness-model/writeup/paper.bib b/correctness-model/writeup/paper.bib new file mode 100644 index 0000000..318493c --- /dev/null +++ b/correctness-model/writeup/paper.bib @@ -0,0 +1,774 @@ +@inproceedings{clockvector, + author = "Friedemann Mattern", + title = "Virtual Time and Global States of Distributed Systems", + booktitle = "Workshop on Parallel and Distributed Algorithms", + year = "1989" +} + +@article{scmemorymodel, + author = {Lamport, Leslie}, + title = { How to Make a Multiprocessor Computer That Correctly Executes Multiprocess Programs}, + journal = {IEEE Transactions on Computers}, + issue_date = {September 1979}, + volume = {28}, + number = {9}, + month = sep, + year = {1979}, + pages = {690--691} +} + +@article{cantin, + author = {Cantin, Jason F. and Lipasti, Mikko H. and Smith, James E.}, + title = {The Complexity of Verifying Memory Coherence and Consistency}, + journal = {IEEE Transactions on Parallel and Distributed Systems}, + issue_date = {July 2005}, + volume = {16}, + number = {7}, + month = jul, + year = {2005}, + issn = {1045-9219}, + pages = {663--671} +} + +@inproceedings{roycav, + author = {Roy, Amitabha and Zeisset, Stephan and Fleckenstein, Charles J. and Huang, John C.}, + title = {Fast and Generalized Polynomial Time Memory Consistency Verification}, + booktitle = {Proceedings of the 18th International Conference on Computer Aided Verification}, + year = {2006} +} + + +@INPROCEEDINGS{qbornotqb, + author = {Ganesh Gopalakrishnan and Yue Yang and Hemanthkumar Sivaraj}, + title = {{QB} or not {QB}: An efficient execution verification tool for memory orderings}, + booktitle = cav04, + year = {2004}, + pages = {401--413} +} + +@Article{testsm, + author = {Phillip B. Gibbons and Ephraim Korach}, + title = {Testing Shared Memories}, + journal = {SIAM Journal on Computing}, + year = {1997}, + volume = {26}, + number = {4}, + pages = {1208-1244}, + month = {August} +} + + + +@inproceedings{koushiksc, + author = {Burnim, Jabob and Sen, Koushik and Stergiou, Christos}, + title = {Sound and complete monitoring of sequential consistency for relaxed memory models}, + booktitle = tacas11, + year = {2011} +} + +@inproceedings{burckhardtverif, + author = {Burckhardt, Sebastian and Musuvathi, Madanlal}, + title = {Effective Program Verification for Relaxed Memory Models}, + booktitle = cav08, + year = {2008} +} + +@inproceedings{checkfence, + author = {Burckhardt, Sebastian and Alur, Rajeev and Martin, Milo M. K.}, + title = {CheckFence: Checking consistency of concurrent data types on relaxed memory models}, + booktitle = pldi07, + year = {2007} +} + +@inproceedings{cdschecker, + author = {Norris, Brian and Demsky, Brian}, + title = {{CDSChecker}: Checking Concurrent Data Structures Written with {C/C++} Atomics}, + booktitle = oopsla13, + year = {2013} +} + +@inproceedings{poplabstraction, + author = {Batty, Mark and Dodds, Mike and Gotsman, Alexey}, + title = {Library abstraction for {C/C++} concurrency}, + booktitle = popl13, + year = {2013} +} + + +@inproceedings{mspcboehm, + author = {Boehm, Hans}, + title = {Can seqlocks get along with programming language memory models?}, + booktitle = mspc12, + year = {2012}, +} + + +@inproceedings{pldisc, + author={Daniel Marino and Abhayendra Singh and Todd Millstein and Madanlal Musuvathi and Satish Narayanasamy}, + title={A Case for an SC-Preserving Compiler}, + booktitle=pldi11, + year={2011} +} + +@inproceedings{ppoppworkstealing, + author = {L\^{e}, Nhat Minh and Pop, Antoniu and Cohen, Albert and Zappa Nardelli, Francesco}, + title = {Correct and efficient work-stealing for weak memory models}, + booktitle = ppopp13, + year = {2013} +} + + +@inproceedings{adversarialmemory, + author = {Flanagan, Cormac and Freund, Stephen N.}, + title = {Adversarial memory for detecting destructive races}, + booktitle = pldi10, + year = {2010} +} + + +@inproceedings{verisoft, + author = {Godefroid, Patrice}, + title = {Model checking for programming languages using {VeriSoft}}, + booktitle = popl97, + year = {1997} +} + + +@ARTICLE{dillmodel, + author = {Seungjoon Park and David L. Dill}, + title = {An Executable Specification and Verifier for Relaxed Memory Order}, + journal = ieeetc, + year = {1999}, + volume = {48} +} + +@inproceedings{vechevpartialcoherence11, + author = {Kuperstein, Michael and Vechev, Martin and Yahav, Eran}, + title = {Partial-coherence abstractions for relaxed memory models}, + booktitle = pldi11, + year = {2011} +} + +@inproceedings{vechevfmcad, + author = {Kuperstein, Michael and Vechev, Martin and Yahav, Eran}, + title = {Automatic inference of memory fences}, + booktitle = fcad10, + year = {2010} +} + + +@article{sparcmodel, + author = {Jonsson, Bengt}, + title = {State-space exploration for concurrent algorithms under weak memory orderings}, + journal = {SIGARCH Computer Architecture News}, + issue_date = {December 2008}, + volume = {36}, + number = {5}, + month = jun, + year = {2009}, + pages = {65--71}, + numpages = {7}, +} + + +@inproceedings{inspect1, + author = {Yang, Yu and Chen, Xiaofang and Gopalakrishnan, Ganesh and Wang, Chao}, + title = {Automatic Discovery of Transition Symmetry in Multithreaded Programs Using Dynamic Analysis}, + booktitle = spin09, + year = {2009}, + location = {Grenoble, France}, + pages = {279--295} +} + + +@article{sleepset, + author = {Godefroid, Patrice}, + title = {Partial-Order Methods for the Verification of Concurrent Systems: An Approach to the State-Explosion Problem}, + year = {1996}, + journal = lncs, + vol = {1032} +} + +@Article{inspect2, + author = {Chao Wang and Yu Yang and Aarti Gupta and Ganesh Gopalakrishnan}, + title = {Dynamic Model Checking with Property Driven Pruning to Detect Race Conditions}, + journal = {ATVA LNCS}, + year={2008}, + number = {126--140} +} + + + +@InProceedings{inspect3, + author = {Yu Yang and Xiaofang Chen and Ganesh Gopalakrishnan and Robert M. Kirby}, + title = {Efficient Stateful Dynamic Partial Order Reduction}, + booktitle = spin08, + year = {2008} +} + + +@inproceedings{inspect4, + author = {Yang, Yu and Chen, Xiaofang and Gopalakrishnan, Ganesh and Kirby, Robert M.}, + title = {Distributed Dynamic Partial Order Reduction Based Verification of Threaded Software}, + booktitle = spin07, + year = {2007}, + pages = {58--75} +} + +@Article{savage1997, + author = {Savage, Stefan and Burrows, Michael and Nelson, Greg and Sobalvarro, Patrick and Anderson, Thomas}, + title = {Eraser: A Dynamic Data Race Detector for Multithreaded Programs}, + journal = tocs, + volume = {15}, + issue = {4}, + month = {Nov.}, + year = {1997}, + pages = {391--411}, +} + +@inproceedings{Zhou2007, + author = {Zhou, Pin and Teodorescu, Radu and Zhou, Yuanyuan}, + title = {{HARD}: Hardware-Assisted Lockset-based Race Detection}, + booktitle = hpca07, + year = {2007}, + pages = {121--132} +} + +@article{Gait1986, + author = {Gait, Jason}, + title = {A probe effect in concurrent programs}, + journal = {Software Practice and Experience}, + volume = {16}, + issue = {3}, + month = {March}, + year = {1986}, + pages = {225--233} +} + + +@inproceedings{Gupta2009, + author = {Gupta, Shantanu and Sultan, Florin and Cadambi, Srihari and Ivancic, Franjo and Rotteler, Martin}, + title = {Using Hardware Transactional Memory for Data Race Detection}, + booktitle = ipdps09, + year = {2009}, + pages = {1--11}, +} + +@inproceedings{Kudrjavets2006, + author = {Kudrjavets, Gunnar and Nagappan, Nachiappan and Ball, Thomas}, + title = {Assessing the Relationship between Software Assertions and Faults: An Empirical Investigation}, + booktitle = {Proceedings of the 17th International Symposium on Software Reliability Engineering}, + year = {2006}, + pages = {204--212} +} + +@inproceedings{lockfreequeue, + author = {Michael, Maged M. and Scott, Michael L.}, + title = {Simple, Fast, and Practical Non-Blocking and Blocking Concurrent Queue Algorithms}, + booktitle = podc96, + year = {1996} +} + +@inproceedings{testera, + author = {Marinov, Darko and Khurshid, Sarfraz}, + title = {TestEra: A Novel Framework for Automated Testing of {Java} Programs}, + booktitle = {Proceedings of the 16th IEEE International Conference on Automated Software Engineering}, + year = {2001} +} + +@article{alloy, + author = {Jackson, Daniel}, + title = {Alloy: A Lightweight Object Modelling Notation}, + journal = {ACM Transactions on Software Engineering and Methodology (TOSEM)}, + volume = {11}, + issue = {2}, + month = {April}, + year = {2002}, + pages = {256--290} +} + + +@InProceedings{ltl, + author = {Amir Pnueli}, + title = {The Temporal Logic of Programs}, + booktitle = {Proceedings of the 18th Annual Symposium on Foundations of Computer Science (FOCS)}, + pages = {46--57}, + Tyear = {1977} +} + +@inproceedings{boehmpldi, + author = {Boehm, Hans J. and Adve, Sarita V.}, + title = {Foundations of the {C++} concurrency memory model}, + booktitle = pldi08, + year = {2008} +} + +@Misc{cpp11spec, + title = {{ISO/IEC 14882:2011}, {Information} Technology -- Programming Languages -- {C++}}, +} + +@Misc{c11spec, + title = {{ISO/IEC 9899:2011}, {Information} Technology -- Programming Languages -- {C}}, +} + +@Misc{ieee1850, + title = {1850-2005 {IEEE} Standard for Property Specification Language ({PSL})}, +} + +@Misc{c1xspec, + title = {N1548: Programming languages -- {C}}, + howpublished = {\url{http://www.open-std.org/jtc1/sc22/wg14/www/docs/n1548.pdf}}, + month = {December}, + year = {2010}, +} + +@InProceedings{fadinew1, + author = {Lu, Shan and Park, Soyeon and Seo, Eunsoo and Zhou, Yuanyuan}, + title = {Learning from Mistakes -- A Comprehensive Study on Real World Concurrency Bug Characteristics}, + booktitle = pldi08, + year = {2008} +} + + +@Article{fadinew2, + author = {Netzer, Robert H. B. and Miller, Barton P.}, + title = {What are Race Conditions?: Some Issues and Formalizations}, + journal = {ACM Letters on Programming Languages and Systems}, + year = {1992}, + volume = {1}, + number = {1}, + pages = {74--88}, + month = {March} +} + +@InProceedings{fadinew3, + author = {Jinpeng Wei and Carlton Pu}, + title = {Multiprocessors May Reduce System Dependability under File-based Race Condition Attacks}, + booktitle = {Proceedings of the 37th Annual IEEE/IFIP International Conference on Dependable Systems and Networks}, + year = {2007} +} + +@Article{rcu, + author = {Mathie Desnoyers and Paul E McKenney and Alan S Stern and Michel R Dagenais and Jonathan Walpole}, + title = {User-Level Implementations of Read-Copy Update}, + journal = tpds, + year = {2011} +} + +@InProceedings{crugiso, + author = {Guoliang Jin and Aditya Thakur and Ben Liblit and + Shan Lu}, + title = {Instrumentation and Sampling Strategies for + {Cooperative} {Concurrency} {Bug} {Isolation}}, + booktitle = oopsla10, + year = 2010 +} + +@inproceedings{incrementalhashing, + author = {Nguyen, Viet Yen and Ruys, Theo C.}, + title = {Incremental Hashing for {SPIN}}, + booktitle = spin08, + year = {2008} +} + +@inproceedings{racetrack, + author = {Yu, Yuan and Rodeheffer, Tom and Chen, Wei}, + title = {RaceTrack: Efficient Detection of Data Race Conditions via Adaptive Tracking}, + booktitle = sosp05, + year = {2005} +} + + +@inproceedings{nitpicking, + author = {Blanchette, Jasmin Christian and Weber, Tjark and Batty, Mark and Owens, Scott and Sarkar, Susmit}, + title = {Nitpicking {C++} concurrency}, + booktitle = ppdp11, + year = {2011} +} + + +@inproceedings{c11popl, + author = {Batty, Mark and Owens, Scott and Sarkar, Susmit and Sewell, Peter and Weber, Tjark}, + title = {Mathematizing {C++} Concurrency}, + booktitle = popl11, + year = {2011} +} + +@inproceedings{fairstateless, + author = {Musuvathi, Madanlal and Qadeer, Shaz}, + title = {Fair stateless model checking}, + booktitle = pldi08, + year = {2008} +} + +@Misc{javaConcurrentHashMap, + author = {Doug Lea}, + title = {util.concurrent.{ConcurrentHashMap} in java.util.concurrent the {Java Concurrency Package}}, + howpublished = {\url{http://docs.oracle.com/javase/8/docs/api/java/util/concurrent/ConcurrentHashMap.html}} +} + +@Misc{clickhashtable, + author = {Cliff Click}, + title = {A Lock-Free Hash Table}, + howpublished = {\url{http://www.azulsystems.com/events/javaone_2007/2007_LockFreeHash.pdf}}, + month = {May}, + year = {2007} +} + +@inproceedings{racerx, + author = {Engler, Dawson and Ashcraft, Ken}, + title = {{RacerX}: Effective, Static Detection of Race Conditions and Deadlocks}, + booktitle = sosp03, + year = {2003} +} + +@inproceedings{conflictexceptions, + author = {Lucia, Brandon and Ceze, Luis and Strauss, Karin and Qadeer, Shaz and Boehm, Hans}, + title = {Conflict Exceptions: Simplifying Concurrent Language Semantics with Precise Hardware Exceptions for Data-Races}, + booktitle = isca10, + year = {2010} +} + + +@inproceedings{fasttrack, + author = {Flanagan, Cormac and Freund, Stephen N.}, + title = {{FastTrack}: Efficient and Precise Dynamic Race Detection}, + booktitle = pldi09, + year = {2009} +} + + +@inproceedings{goldilocks, + author = {Elmas, Tayfun and Qadeer, Shaz and Tasiran, Serdar}, + title = {Goldilocks: A Race and Transaction-Aware {Java} Runtime}, + booktitle = pldi07, + year = {2007} +} + +@inproceedings{boyapati, + author = {Chandrasekhar Boyapati and Robert Lee and Martin Rinard}, + title = {{Ownership Types for Safe Programming: Preventing Data Races and Deadlocks}}, + booktitle = oopsla02, + year = {2002} + } + + +@Misc{relacy, + author = {Dmitriy Vyukov}, + title = {Relacy Race Detector}, + howpublished = {\url{http://relacy.sourceforge.net/}}, + month = {2011}, + year = {Oct.} +} + +@Book{spinmodel, + author = {Gerard J. Holzmann}, + title = {The {SPIN} Model Checker: Primer and Reference Manual}, + publisher = {Addison-Wesley Professional}, + year = {2003}, + edition = {1st} +} + +@Book{multiprocessorprogramming, + author = {Maurice Herlihy and Nir Shavit}, + title = {The Art of Multiprocessor Programming}, + publisher = {Morgan Kaufmamn}, + year = {2012}, + edition = {Revised 1st} +} + +@InProceedings{dpor, + author = {Cormac Flanagan and Patrice Godefroid}, + title = {Dynamic Partial-Order Reduction for Model Checking Software}, + booktitle = popl05, + month = {Jan.}, + year = {2005} +} + +@Article{shavitcacm, + author = {Nir Shavit}, + title = {Data Structures in the Multicore Age}, + journal = cacm, + year = {2011}, + volume = {54}, + number = {3}, + month = {March} +} +% pages = {76--84}, + +@inproceedings{chess, + author = {Madanlal Musuvathi and Shaz Qadeer and Piramanayagam Arumuga Nainar and Thomas Ball and Gerard Basler and Iulian Neamtiu}, + title = {Finding and Reproducing {Heisenbugs} in Concurrent Programs}, + year = {2008}, + booktitle=osdi08 +} + + +@inproceedings{lineup, + author = {Burckhardt, Sebastian and Dern, Chris and Musuvathi, Madanlal and Tan, Roy}, + title = {Line-up: A Complete and Automatic Linearizability Checker}, + booktitle = pldi10, + year = {2010} +} + +@inproceedings{berger-grace-09, + author = {Berger, Emery D. and Yang, Ting and Liu, Tongping and Novark, Gene}, + title = {Grace: Safe multithreaded programming for {C}/{C}++}, + booktitle = oopsla09, + year = {2009}, +} + +@INPROCEEDINGS{charisma, +author = {Chao Huang and Laxmikant V. Kale}, +title = {Charisma: Orchestrating Migratable Parallel Objects}, +booktitle = hpdc07, +year = {2007} +} + + +@inproceedings{ding-bop-07, + author = {Ding, Chen and Shen, Xipeng and Kelsey, Kirk and Tice, Chris and Huang, Ruke and Zhang, Chengliang}, + title = {Software behavior oriented parallelization}, + booktitle = pldi07, + year = {2007}, +} + +@Misc{tilepro64, + key = {tilera}, + title = {TILEPro64 Processor}, + howpublished = {\url{http://tilera.com/products/processors/TILEPRO64}}, + month = {December}, + year = {2010} +} + +@Misc{magnycour, + key = {amd}, + title = {{AMD} {Opteron} 6000 Series Platform}, + howpublished = {\url{http://www.amd.com/us/products/server/processors/6000-series-platform/pages/6000-series-platform.aspx}}, + month = {December}, + year = {2010} +} + +@inproceedings{delaunay, + author = {Guibas, Leonidas J. and Knuth, Donald E. and Sharir, Micha}, + title = {Randomized Incremental Construction of {Delaunay} and {Voronoi} Diagrams}, + booktitle = {Proceedings of the Seventeenth International Colloquium on Automata, Languages and Programming}, + year = {1990}, + location = {Warwick University, England}, + pages = {414--431}, + numpages = {18}, + publisher = {Springer-Verlag New York, Inc.}, + address = {New York, NY, USA}, +} + +@inproceedings{1000core, + author = {Borkar, Shekhar}, + title = {Thousand Core Chips: A Technology Perspective}, + booktitle = dac07, + year = {2007} + } + +@PhDThesis{r-cilk-98, + author = {Keith H. Randall}, + title = {Cilk: Efficient Multithreaded Computing}, + school = {Massachusetts Institute of Technology}, + year = 1998, +} + +@InProceedings{dll-jcilk-05, + author = {John S. Danaher and I-Ting Angelina Lee and Charles E. Leiserson}, + title = {The {JCilk} Language for Multithreaded Computing}, + booktitle = scool05, + year = 2005, +} + +@inproceedings{java-grande, + author = {Smith, L. A. and Bull, J. M. and Obdrz\'{a}lek, J.}, + title = {A Parallel {J}ava {G}rande Benchmark Suite}, + booktitle = sc01, + year = {2001} +} + +@inproceedings{spin-commit-atomicity, + author = {Flanagan, Cormac}, + title = {Verifying Commit-Atomicity using Model-Checking}, + booktitle = spin04, + year = {2004} +} + +@Article{linearizableref, + author = {Maurice Herlihy and Jeannette Wing}, + title = {Linearizability: a correctness condition for concurrent objects}, + journal = {ACM Transactions on Programming Languages and Systems}, + year = {1990}, + volume = {12}, + number = {3}, + pages = {463-492}, + month = {July} +} + +@Article{refinement-mapping, + author = {Abadi, Martine and Lamport, Leslie}, + title = {The existence of Refinement Mapping}, + journal = {Theoretical Computer Science}, + year = {1991}, + volume = {82}, + number = {2}, + pages = {253-284}, + month = {May} +} + +@inproceedings{abstraction-linearizability, + author = {Amit, D. and Rinetzky, N. and Reps, T. and Sagiv, M. and Yahav, E. }, + title = {Comparison under Abstraction for Verifying Linearizability}, + booktitle = cav07, + year = {2007} +} + +@inproceedings{formal-verification-set, + author = {Colvin, R. and Groves, L. and Luchangco, V. and Moir, M.}, + title = {Formal Verification of a Lazy Concurrent List-Based Set Algorithm}, + booktitle = cav06, + year = {2006} +} + +@inproceedings{concurrit, + author = {Elmas, T. and Burnim, J. and Necula, G. and Sen, K.}, + title = {{CONCURRIT}: A Domain Specific Language for Reproducing Concurrency Bugs}, + booktitle = pldi13, + year = {2013} +} + +@inproceedings{gambit, + author = {Coons, K. E. and Burckhardt, S. and Musuvathi, M.}, + title = {{GAMBIT}: Effective Unit Testing for Concurrency Libraries}, + booktitle = ppopp10, + year = {2010} +} + +@inproceedings{memtest, + author = {Burnim, J. and Sen, K. and Stergiou, C.}, + title = {Testing Concurrent Programs on Relaxed Memory Models}, + booktitle = issta11, + year = {2011} +} + +@inproceedings{ndetermin, + author = {Burnim, J. and Elmas, T. and Necula, G. and Sen, K.}, + title = {{NDetermin}: Inferring Nondeterministic Sequential Specifications for Parallelism Correctness}, + booktitle = ppopp12, + year = {2012} +} + +@inproceedings{vyrd, + author = {Elmas, T. and Tasiran, S. and Qadeer, S.}, + title = {{VYRD}: VerifYing Concurrent Programs by Runtime Refinement-Violation Detection}, + booktitle = pldi05, + year = {2005} +} + +@inproceedings{thread-quantification, + author = {Berdine, J. and Lev-Ami, T. and Manivich, R. and Ramalingam, G. and Sagiv, M.}, + title = {Thread Quantification for Concurrent Shape Analysis}, + booktitle = cav08, + year = {2008} +} + +@inproceedings{VechevMCLinear, + author = {Vechev, Martin and Yahav, Eran and Yorsh, Greta}, + title = {Experience with Model Checking Linearizability}, + booktitle = "International SPIN Workshop on Model Checking Software", + year = {2009} +} + +@inproceedings{shape-value-abstraction, + author = {Vafeiadis, Viktor}, + title = {Shape-Value Abstraction for Verifying Linearizability}, + booktitle = vmcai09, + year = {2009} +} + +@Article{test-verify-concurrent-objects, + author = {Wing, Jeannette M. and Gong, Chun}, + title = {Testing and Verifying Concurrent Objects}, + journal = {Journal of Parallel and Distributed Computing - Special issue on parallel I/O systems}, + year = {1993}, + volume = {17}, + number = {1-2}, + pages = {164-182}, + month = {Jan./Feb.} +} + +@inproceedings{stamp, + title = {{STAMP}: Stanford Transactional Applications for Multi-Processing}, + author = {Cao Minh, Chi and Chung, JaeWoong and Kozyrakis, Christos and Olukotun, Kunle}, + booktitle = iiswc08, + year = {2008} +} + +@article{l-clocks, + author = {Lamport, Leslie}, + title = {Time, clocks, and the ordering of events in a distributed system}, + journal = cacm, + issue_date = {July 1978}, + volume = {21}, + number = {7}, + month = jul, + year = {1978}, + pages = {558--565}, + numpages = {8}, + acmid = {359563}, + publisher = {ACM}, + address = {New York, NY, USA}, + keywords = {clock synchronization, computer networks, distributed systems, multiprocess systems}, +} + +@inproceedings{reverse-execution, + author = {Pan, Douglas Z. and Linton, Mark A.}, + title = {Supporting reverse execution for parallel programs}, + booktitle = padd88, + year = {1988} +} + +@misc{linux, + title = {Linux Kernel v3.6}, + howpublished = {\url{http://kernel.org/}}, + month = {September}, + year = {2012}, +} + +@inproceedings{mcs-lock, + author = {Mellor-Crummey, John M. and Scott, Michael L.}, + title = {Synchronization without contention}, + booktitle = asplos91, + year = {1991}, + location = {Santa Clara, California, United States}, + pages = {269--278}, + numpages = {10}, + acmid = {106999}, +} + +@misc{barrier-url, + howpublished = {\url{http://stackoverflow.com/questions/8115267/writing-a-spinning-thread-barrier-using-c11-atomics}}, + note = {Oct. 2012}, +} + +@misc{mcs-lock-url, + howpublished = {\url{http://cbloomrants.blogspot.com/2011/07/07-18-11-mcs-list-based-lock\_18.html}}, + note = {Oct. 2012}, +} + +@misc{spsc-queue-url, + howpublished = {\url{https://groups.google.com/forum/#!msg/comp.programming.threads/nSSFT9vKEe0/7eD3ioDg6nEJ}}, + note = {Oct. 2012}, +} + +@misc{mpmc-queue-url, + howpublished = {\url{http://cbloomrants.blogspot.com/2011/07/07-30-11-look-at-some-bounded-queues.html}}, + note = {Oct. 2012}, +} + +@misc{dekker-url, + howpublished = {\url{http://www.justsoftwaresolutions.co.uk/threading/}}, + note = {Dec. 2012}, +} diff --git a/correctness-model/writeup/paper.blg b/correctness-model/writeup/paper.blg new file mode 100644 index 0000000..6a4c408 --- /dev/null +++ b/correctness-model/writeup/paper.blg @@ -0,0 +1,50 @@ +This is BibTeX, Version 0.99d (TeX Live 2013/Debian) +Capacity: max_strings=35307, hash_size=35307, hash_prime=30011 +The top-level auxiliary file: paper.aux +The style file: abbrv.bst +Database file #1: confstrs-long.bib +Database file #2: paper.bib +Warning--to sort, need author or key in cpp11spec +Warning--to sort, need author or key in c11spec +You've used 14 entries, + 2118 wiz_defined-function locations, + 816 strings with 16605 characters, +and the built_in function-call counts, 3810 in all, are: += -- 331 +> -- 190 +< -- 0 ++ -- 76 +- -- 62 +* -- 224 +:= -- 631 +add.period$ -- 38 +call.type$ -- 14 +change.case$ -- 73 +chr.to.int$ -- 0 +cite$ -- 16 +duplicate$ -- 158 +empty$ -- 358 +format.name$ -- 62 +if$ -- 806 +int.to.chr$ -- 0 +int.to.str$ -- 14 +missing$ -- 10 +newline$ -- 69 +num.names$ -- 24 +pop$ -- 115 +preamble$ -- 1 +purify$ -- 59 +quote$ -- 0 +skip$ -- 108 +stack$ -- 0 +substring$ -- 116 +swap$ -- 21 +text.length$ -- 0 +text.prefix$ -- 0 +top$ -- 0 +type$ -- 56 +warning$ -- 2 +while$ -- 27 +width$ -- 16 +write$ -- 133 +(There were 2 warnings) diff --git a/correctness-model/writeup/paper.log b/correctness-model/writeup/paper.log new file mode 100644 index 0000000..7b65903 --- /dev/null +++ b/correctness-model/writeup/paper.log @@ -0,0 +1,661 @@ +This is pdfTeX, Version 3.1415926-2.5-1.40.14 (TeX Live 2013/Debian) (format=pdflatex 2014.3.31) 10 APR 2015 10:40 +entering extended mode + restricted \write18 enabled. + %&-line parsing enabled. +**paper.tex +(./paper.tex +LaTeX2e <2011/06/27> +Babel <3.9h> and hyphenation patterns for 78 languages loaded. +(./sig-alternate.cls +(/usr/share/texlive/texmf-dist/tex/latex/graphics/epsfig.sty +Package: epsfig 1999/02/16 v1.7a (e)psfig emulation (SPQR) + +(/usr/share/texlive/texmf-dist/tex/latex/graphics/graphicx.sty +Package: graphicx 1999/02/16 v1.0f Enhanced LaTeX Graphics (DPC,SPQR) + +(/usr/share/texlive/texmf-dist/tex/latex/graphics/keyval.sty +Package: keyval 1999/03/16 v1.13 key=value parser (DPC) +\KV@toks@=\toks14 +) +(/usr/share/texlive/texmf-dist/tex/latex/graphics/graphics.sty +Package: graphics 2009/02/05 v1.0o Standard LaTeX Graphics (DPC,SPQR) + +(/usr/share/texlive/texmf-dist/tex/latex/graphics/trig.sty +Package: trig 1999/03/16 v1.09 sin cos tan (DPC) +) +(/etc/texmf/tex/latex/config/graphics.cfg +File: graphics.cfg 2009/08/28 v1.8 graphics configuration of TeX Live +) +Package graphics Info: Driver file: pdftex.def on input line 91. + +(/usr/share/texlive/texmf-dist/tex/latex/pdftex-def/pdftex.def +File: pdftex.def 2011/05/27 v0.06d Graphics/color for pdfTeX + +(/usr/share/texlive/texmf-dist/tex/generic/oberdiek/infwarerr.sty +Package: infwarerr 2010/04/08 v1.3 Providing info/warning/error messages (HO) +) +(/usr/share/texlive/texmf-dist/tex/generic/oberdiek/ltxcmds.sty +Package: ltxcmds 2011/11/09 v1.22 LaTeX kernel commands for general use (HO) +) +\Gread@gobject=\count79 +)) +\Gin@req@height=\dimen102 +\Gin@req@width=\dimen103 +) +\epsfxsize=\dimen104 +\epsfysize=\dimen105 +) +(/usr/share/texlive/texmf-dist/tex/latex/amsfonts/amssymb.sty +Package: amssymb 2013/01/14 v3.01 AMS font symbols + +(/usr/share/texlive/texmf-dist/tex/latex/amsfonts/amsfonts.sty +Package: amsfonts 2013/01/14 v3.01 Basic AMSFonts support +\@emptytoks=\toks15 +\symAMSa=\mathgroup4 +\symAMSb=\mathgroup5 +LaTeX Font Info: Overwriting math alphabet `\mathfrak' in version `bold' +(Font) U/euf/m/n --> U/euf/b/n on input line 106. +)) +(/usr/share/texlive/texmf-dist/tex/latex/amsmath/amsmath.sty +Package: amsmath 2013/01/14 v2.14 AMS math features +\@mathmargin=\skip41 + +For additional information on amsmath, use the `?' option. +(/usr/share/texlive/texmf-dist/tex/latex/amsmath/amstext.sty +Package: amstext 2000/06/29 v2.01 + +(/usr/share/texlive/texmf-dist/tex/latex/amsmath/amsgen.sty +File: amsgen.sty 1999/11/30 v2.0 +\@emptytoks=\toks16 +\ex@=\dimen106 +)) +(/usr/share/texlive/texmf-dist/tex/latex/amsmath/amsbsy.sty +Package: amsbsy 1999/11/29 v1.2d +\pmbraise@=\dimen107 +) +(/usr/share/texlive/texmf-dist/tex/latex/amsmath/amsopn.sty +Package: amsopn 1999/12/14 v2.01 operator names +) +\inf@bad=\count80 +LaTeX Info: Redefining \frac on input line 210. +\uproot@=\count81 +\leftroot@=\count82 +LaTeX Info: Redefining \overline on input line 306. +\classnum@=\count83 +\DOTSCASE@=\count84 +LaTeX Info: Redefining \ldots on input line 378. +LaTeX Info: Redefining \dots on input line 381. +LaTeX Info: Redefining \cdots on input line 466. +\Mathstrutbox@=\box26 +\strutbox@=\box27 +\big@size=\dimen108 +LaTeX Font Info: Redeclaring font encoding OML on input line 566. +LaTeX Font Info: Redeclaring font encoding OMS on input line 567. +\macc@depth=\count85 +\c@MaxMatrixCols=\count86 +\dotsspace@=\muskip10 +\c@parentequation=\count87 +\dspbrk@lvl=\count88 +\tag@help=\toks17 +\row@=\count89 +\column@=\count90 +\maxfields@=\count91 +\andhelp@=\toks18 +\eqnshift@=\dimen109 +\alignsep@=\dimen110 +\tagshift@=\dimen111 +\tagwidth@=\dimen112 +\totwidth@=\dimen113 +\lineht@=\dimen114 +\@envbody=\toks19 +\multlinegap=\skip42 +\multlinetaggap=\skip43 +\mathdisplay@stack=\toks20 +LaTeX Info: Redefining \[ on input line 2665. +LaTeX Info: Redefining \] on input line 2666. +) +Document Class 'sig-alternate' <23rd. May '12>. Modified by G.K.M. Tobin/Gerry +Murray +Based in part upon document Style `acmconf' <22 May 89>. Hacked 4/91 by +shivers@cs.cmu.edu, 4/93 by theobald@cs.mcgill.ca +Excerpts were taken from (Journal Style) 'esub2acm.cls'. +****** Bugs/comments/suggestions/technicalities to Gerry Murray -- murray@hq.ac +m.org ****** +Questions on the style, SIGS policies, etc. to Adrienne Griscti griscti@acm.org + +\footheight=\dimen115 +\@maxsep=\dimen116 +\@dblmaxsep=\dimen117 +\aucount=\count92 +\originalaucount=\count93 +\auwidth=\dimen118 +\auskip=\dimen119 +\auskipcount=\count94 +\auskip=\dimen120 +\allauboxes=\dimen121 +\addauthors=\toks21 +\addauflag=\count95 +\subtitletext=\toks22 +\savesection=\count96 +\sectioncntr=\count97 +\c@figure=\count98 +\c@table=\count99 +\titleboxnotes=\toks23 +\titleboxnoteflag=\count100 +Document Class: sig-alternate 2012/05/23 - V2.5 - based on acmproc.cls V1.3 +(/usr/share/texlive/texmf-dist/tex/latex/base/latexsym.sty +Package: latexsym 1998/08/17 v2.2e Standard LaTeX package (lasy symbols) +\symlasy=\mathgroup6 +LaTeX Font Info: Overwriting symbol font `lasy' in version `bold' +(Font) U/lasy/m/n --> U/lasy/b/n on input line 47. +) +\@acmtitlebox=\box28 +\titlenotecount=\count101 +\tntoks=\toks24 +\tntokstwo=\toks25 +\tntoksthree=\toks26 +\tntoksfour=\toks27 +\tntoksfive=\toks28 +\catcount=\count102 +\copyrightnotice=\toks29 +\conf=\toks30 +\confinfo=\toks31 +\c@part=\count103 +\c@section=\count104 +\c@subsection=\count105 +\c@subsubsection=\count106 +\c@paragraph=\count107 + +Using 'Abbrev' bibliography style +LaTeX Info: Redefining \cite on input line 1211. +\bibindent=\dimen122 +\colcntr=\count108 +\saveb@x=\box29 +\copyrtyr=\toks32 +\acmcopyr=\toks33 +\boilerplate=\toks34 +\copyrightetc=\toks35 +(/usr/share/texlive/texmf-dist/tex/latex/base/fontenc.sty +Package: fontenc 2005/09/27 v1.99g Standard LaTeX package + +(/usr/share/texlive/texmf-dist/tex/latex/base/t1enc.def +File: t1enc.def 2005/09/27 v1.99g Standard LaTeX file +LaTeX Font Info: Redeclaring font encoding T1 on input line 43. +) +LaTeX Font Info: Try loading font information for T1+aer on input line 100. + +(/usr/share/texlive/texmf-dist/tex/latex/ae/t1aer.fd +File: t1aer.fd 1997/11/16 Font definitions for T1/aer. +))) +(/usr/share/texlive/texmf-dist/tex/latex/psnfss/times.sty +Package: times 2005/04/12 PSNFSS-v9.2a (SPQR) +) +(/usr/share/texlive/texmf-dist/tex/latex/psnfss/helvet.sty +Package: helvet 2005/04/12 PSNFSS-v9.2a (WaS) +) +(/usr/share/texlive/texmf-dist/tex/latex/url/url.sty +\Urlmuskip=\muskip11 +Package: url 2013/09/16 ver 3.4 Verb mode for urls, etc. +) +(/usr/share/texlive/texmf-dist/tex/latex/listings/listings.sty +\lst@mode=\count109 +\lst@gtempboxa=\box30 +\lst@token=\toks36 +\lst@length=\count110 +\lst@currlwidth=\dimen123 +\lst@column=\count111 +\lst@pos=\count112 +\lst@lostspace=\dimen124 +\lst@width=\dimen125 +\lst@newlines=\count113 +\lst@lineno=\count114 +\abovecaptionskip=\skip44 +\belowcaptionskip=\skip45 +\lst@maxwidth=\dimen126 + +(/usr/share/texlive/texmf-dist/tex/latex/listings/lstmisc.sty +File: lstmisc.sty 2013/08/26 1.5b (Carsten Heinz) +\c@lstnumber=\count115 +\lst@skipnumbers=\count116 +\lst@framebox=\box31 +) +(/usr/share/texlive/texmf-dist/tex/latex/listings/listings.cfg +File: listings.cfg 2013/08/26 1.5b listings configuration +)) +Package: listings 2013/08/26 1.5b (Carsten Heinz) + +(/usr/share/texlive/texmf-dist/tex/latex/enumitem/enumitem.sty +Package: enumitem 2011/09/28 v3.5.2 Customized lists +\labelindent=\skip46 +\enit@outerparindent=\dimen127 +\enit@toks=\toks37 +\enit@inbox=\box32 +\enitdp@description=\count117 +) +(/usr/share/texlive/texmf-dist/tex/latex/hyperref/hyperref.sty +Package: hyperref 2012/11/06 v6.83m Hypertext links for LaTeX + +(/usr/share/texlive/texmf-dist/tex/generic/oberdiek/hobsub-hyperref.sty +Package: hobsub-hyperref 2012/05/28 v1.13 Bundle oberdiek, subset hyperref (HO) + + +(/usr/share/texlive/texmf-dist/tex/generic/oberdiek/hobsub-generic.sty +Package: hobsub-generic 2012/05/28 v1.13 Bundle oberdiek, subset generic (HO) +Package: hobsub 2012/05/28 v1.13 Construct package bundles (HO) +Package hobsub Info: Skipping package `infwarerr' (already loaded). +Package hobsub Info: Skipping package `ltxcmds' (already loaded). +Package: ifluatex 2010/03/01 v1.3 Provides the ifluatex switch (HO) +Package ifluatex Info: LuaTeX not detected. +Package: ifvtex 2010/03/01 v1.5 Detect VTeX and its facilities (HO) +Package ifvtex Info: VTeX not detected. +Package: intcalc 2007/09/27 v1.1 Expandable calculations with integers (HO) +Package: ifpdf 2011/01/30 v2.3 Provides the ifpdf switch (HO) +Package ifpdf Info: pdfTeX in PDF mode is detected. +Package: etexcmds 2011/02/16 v1.5 Avoid name clashes with e-TeX commands (HO) +Package etexcmds Info: Could not find \expanded. +(etexcmds) That can mean that you are not using pdfTeX 1.50 or +(etexcmds) that some package has redefined \expanded. +(etexcmds) In the latter case, load this package earlier. +Package: kvsetkeys 2012/04/25 v1.16 Key value parser (HO) +Package: kvdefinekeys 2011/04/07 v1.3 Define keys (HO) +Package: pdftexcmds 2011/11/29 v0.20 Utility functions of pdfTeX for LuaTeX (HO +) +Package pdftexcmds Info: LuaTeX not detected. +Package pdftexcmds Info: \pdf@primitive is available. +Package pdftexcmds Info: \pdf@ifprimitive is available. +Package pdftexcmds Info: \pdfdraftmode found. +Package: pdfescape 2011/11/25 v1.13 Implements pdfTeX's escape features (HO) +Package: bigintcalc 2012/04/08 v1.3 Expandable calculations on big integers (HO +) +Package: bitset 2011/01/30 v1.1 Handle bit-vector datatype (HO) +Package: uniquecounter 2011/01/30 v1.2 Provide unlimited unique counter (HO) +) +Package hobsub Info: Skipping package `hobsub' (already loaded). +Package: letltxmacro 2010/09/02 v1.4 Let assignment for LaTeX macros (HO) +Package: hopatch 2012/05/28 v1.2 Wrapper for package hooks (HO) +Package: xcolor-patch 2011/01/30 xcolor patch +Package: atveryend 2011/06/30 v1.8 Hooks at the very end of document (HO) +Package atveryend Info: \enddocument detected (standard20110627). +Package: atbegshi 2011/10/05 v1.16 At begin shipout hook (HO) +Package: refcount 2011/10/16 v3.4 Data extraction from label references (HO) +Package: hycolor 2011/01/30 v1.7 Color options for hyperref/bookmark (HO) +) +(/usr/share/texlive/texmf-dist/tex/generic/ifxetex/ifxetex.sty +Package: ifxetex 2010/09/12 v0.6 Provides ifxetex conditional +) +(/usr/share/texlive/texmf-dist/tex/latex/oberdiek/auxhook.sty +Package: auxhook 2011/03/04 v1.3 Hooks for auxiliary files (HO) +) +(/usr/share/texlive/texmf-dist/tex/latex/oberdiek/kvoptions.sty +Package: kvoptions 2011/06/30 v3.11 Key value format for package options (HO) +) +\@linkdim=\dimen128 +\Hy@linkcounter=\count118 +\Hy@pagecounter=\count119 + +(/usr/share/texlive/texmf-dist/tex/latex/hyperref/pd1enc.def +File: pd1enc.def 2012/11/06 v6.83m Hyperref: PDFDocEncoding definition (HO) +) +\Hy@SavedSpaceFactor=\count120 + +(/usr/share/texlive/texmf-dist/tex/latex/latexconfig/hyperref.cfg +File: hyperref.cfg 2002/06/06 v1.2 hyperref configuration of TeXLive +) +Package hyperref Info: Option `colorlinks' set `true' on input line 4319. +Package hyperref Info: Option `breaklinks' set `true' on input line 4319. +Package hyperref Info: Option `draft' set `true' on input line 4319. +Package hyperref Info: Hyper figures OFF on input line 4443. +Package hyperref Info: Link nesting OFF on input line 4448. +Package hyperref Info: Hyper index ON on input line 4451. +Package hyperref Info: Plain pages OFF on input line 4458. +Package hyperref Info: Backreferencing OFF on input line 4463. +Package hyperref Info: Implicit mode ON; LaTeX internals redefined. +Package hyperref Info: Bookmarks ON on input line 4688. +\c@Hy@tempcnt=\count121 +LaTeX Info: Redefining \url on input line 5041. +\XeTeXLinkMargin=\dimen129 +\Fld@menulength=\count122 +\Field@Width=\dimen130 +\Fld@charsize=\dimen131 +Package hyperref Info: Hyper figures OFF on input line 6295. +Package hyperref Info: Link nesting OFF on input line 6300. +Package hyperref Info: Hyper index ON on input line 6303. +Package hyperref Info: backreferencing OFF on input line 6310. +Package hyperref Info: Link coloring ON on input line 6313. +Package hyperref Info: Link coloring with OCG OFF on input line 6320. +Package hyperref Info: PDF/A mode OFF on input line 6325. +LaTeX Info: Redefining \ref on input line 6365. +LaTeX Info: Redefining \pageref on input line 6369. + + +Package hyperref Warning: Option `pdfpagelabels' is turned off +(hyperref) because \thepage is undefined. + +\c@Item=\count123 +\c@Hfootnote=\count124 +) + +Package hyperref Message: Driver (autodetected): hpdftex. + +(/usr/share/texlive/texmf-dist/tex/latex/hyperref/hpdftex.def +File: hpdftex.def 2012/11/06 v6.83m Hyperref driver for pdfTeX +\Fld@listcount=\count125 +\c@bookmark@seq@number=\count126 + +(/usr/share/texlive/texmf-dist/tex/latex/oberdiek/rerunfilecheck.sty +Package: rerunfilecheck 2011/04/15 v1.7 Rerun checks for auxiliary files (HO) +Package uniquecounter Info: New unique counter `rerunfilecheck' on input line 2 +82. +) +\Hy@SectionHShift=\skip47 +) +(/usr/share/texlive/texmf-dist/tex/latex/rotating/rotating.sty +Package: rotating 2009/03/28 v2.16a rotated objects in LaTeX + +(/usr/share/texlive/texmf-dist/tex/latex/base/ifthen.sty +Package: ifthen 2001/05/26 v1.1c Standard LaTeX ifthen package (DPC) +) +\c@r@tfl@t=\count127 +\rotFPtop=\skip48 +\rotFPbot=\skip49 +\rot@float@box=\box33 +\rot@mess@toks=\toks38 +) +(/usr/share/texlive/texmf-dist/tex/latex/algorithms/algorithm.sty +Package: algorithm 2009/08/24 v0.1 Document Style `algorithm' - floating enviro +nment + +(/usr/share/texlive/texmf-dist/tex/latex/float/float.sty +Package: float 2001/11/08 v1.3d Float enhancements (AL) +\c@float@type=\count128 +\float@exts=\toks39 +\float@box=\box34 +\@float@everytoks=\toks40 +\@floatcapt=\box35 +) +\@float@every@algorithm=\toks41 +\c@algorithm=\count129 +) +(/usr/share/texlive/texmf-dist/tex/latex/algorithmicx/algpseudocode.sty +Package: algpseudocode + +(/usr/share/texlive/texmf-dist/tex/latex/algorithmicx/algorithmicx.sty +Package: algorithmicx 2005/04/27 v1.2 Algorithmicx + +Document Style algorithmicx 1.2 - a greatly improved `algorithmic' style +\c@ALG@line=\count130 +\c@ALG@rem=\count131 +\c@ALG@nested=\count132 +\ALG@tlm=\skip50 +\ALG@thistlm=\skip51 +\c@ALG@Lnr=\count133 +\c@ALG@blocknr=\count134 +\c@ALG@storecount=\count135 +\c@ALG@tmpcounter=\count136 +\ALG@tmplength=\skip52 +) +Document Style - pseudocode environments for use with the `algorithmicx' style +) (/usr/share/texlive/texmf-dist/tex/latex/tools/xspace.sty +Package: xspace 2009/10/20 v1.13 Space after command names (DPC,MH) +) +(/usr/share/texlive/texmf-dist/tex/latex/graphics/color.sty +Package: color 2005/11/14 v1.0j Standard LaTeX Color (DPC) + +(/etc/texmf/tex/latex/config/color.cfg +File: color.cfg 2007/01/18 v1.5 color configuration of teTeX/TeXLive +) +Package color Info: Driver file: pdftex.def on input line 130. +) +(/usr/share/texlive/texmf-dist/tex/latex/base/textcomp.sty +Package: textcomp 2005/09/27 v1.99g Standard LaTeX package +Package textcomp Info: Sub-encoding information: +(textcomp) 5 = only ISO-Adobe without \textcurrency +(textcomp) 4 = 5 + \texteuro +(textcomp) 3 = 4 + \textohm +(textcomp) 2 = 3 + \textestimated + \textcurrency +(textcomp) 1 = TS1 - \textcircled - \t +(textcomp) 0 = TS1 (full) +(textcomp) Font families with sub-encoding setting implement +(textcomp) only a restricted character set as indicated. +(textcomp) Family '?' is the default used for unknown fonts. +(textcomp) See the documentation for details. +Package textcomp Info: Setting ? sub-encoding to TS1/1 on input line 71. + +(/usr/share/texlive/texmf-dist/tex/latex/base/ts1enc.def +File: ts1enc.def 2001/06/05 v3.0e (jk/car/fm) Standard LaTeX file +) +LaTeX Info: Redefining \oldstylenums on input line 266. +Package textcomp Info: Setting cmr sub-encoding to TS1/0 on input line 281. +Package textcomp Info: Setting cmss sub-encoding to TS1/0 on input line 282. +Package textcomp Info: Setting cmtt sub-encoding to TS1/0 on input line 283. +Package textcomp Info: Setting cmvtt sub-encoding to TS1/0 on input line 284. +Package textcomp Info: Setting cmbr sub-encoding to TS1/0 on input line 285. +Package textcomp Info: Setting cmtl sub-encoding to TS1/0 on input line 286. +Package textcomp Info: Setting ccr sub-encoding to TS1/0 on input line 287. +Package textcomp Info: Setting ptm sub-encoding to TS1/4 on input line 288. +Package textcomp Info: Setting pcr sub-encoding to TS1/4 on input line 289. +Package textcomp Info: Setting phv sub-encoding to TS1/4 on input line 290. +Package textcomp Info: Setting ppl sub-encoding to TS1/3 on input line 291. +Package textcomp Info: Setting pag sub-encoding to TS1/4 on input line 292. +Package textcomp Info: Setting pbk sub-encoding to TS1/4 on input line 293. +Package textcomp Info: Setting pnc sub-encoding to TS1/4 on input line 294. +Package textcomp Info: Setting pzc sub-encoding to TS1/4 on input line 295. +Package textcomp Info: Setting bch sub-encoding to TS1/4 on input line 296. +Package textcomp Info: Setting put sub-encoding to TS1/5 on input line 297. +Package textcomp Info: Setting uag sub-encoding to TS1/5 on input line 298. +Package textcomp Info: Setting ugq sub-encoding to TS1/5 on input line 299. +Package textcomp Info: Setting ul8 sub-encoding to TS1/4 on input line 300. +Package textcomp Info: Setting ul9 sub-encoding to TS1/4 on input line 301. +Package textcomp Info: Setting augie sub-encoding to TS1/5 on input line 302. +Package textcomp Info: Setting dayrom sub-encoding to TS1/3 on input line 303. +Package textcomp Info: Setting dayroms sub-encoding to TS1/3 on input line 304. + +Package textcomp Info: Setting pxr sub-encoding to TS1/0 on input line 305. +Package textcomp Info: Setting pxss sub-encoding to TS1/0 on input line 306. +Package textcomp Info: Setting pxtt sub-encoding to TS1/0 on input line 307. +Package textcomp Info: Setting txr sub-encoding to TS1/0 on input line 308. +Package textcomp Info: Setting txss sub-encoding to TS1/0 on input line 309. +Package textcomp Info: Setting txtt sub-encoding to TS1/0 on input line 310. +Package textcomp Info: Setting lmr sub-encoding to TS1/0 on input line 311. +Package textcomp Info: Setting lmdh sub-encoding to TS1/0 on input line 312. +Package textcomp Info: Setting lmss sub-encoding to TS1/0 on input line 313. +Package textcomp Info: Setting lmssq sub-encoding to TS1/0 on input line 314. +Package textcomp Info: Setting lmvtt sub-encoding to TS1/0 on input line 315. +Package textcomp Info: Setting qhv sub-encoding to TS1/0 on input line 316. +Package textcomp Info: Setting qag sub-encoding to TS1/0 on input line 317. +Package textcomp Info: Setting qbk sub-encoding to TS1/0 on input line 318. +Package textcomp Info: Setting qcr sub-encoding to TS1/0 on input line 319. +Package textcomp Info: Setting qcs sub-encoding to TS1/0 on input line 320. +Package textcomp Info: Setting qpl sub-encoding to TS1/0 on input line 321. +Package textcomp Info: Setting qtm sub-encoding to TS1/0 on input line 322. +Package textcomp Info: Setting qzc sub-encoding to TS1/0 on input line 323. +Package textcomp Info: Setting qhvc sub-encoding to TS1/0 on input line 324. +Package textcomp Info: Setting futs sub-encoding to TS1/4 on input line 325. +Package textcomp Info: Setting futx sub-encoding to TS1/4 on input line 326. +Package textcomp Info: Setting futj sub-encoding to TS1/4 on input line 327. +Package textcomp Info: Setting hlh sub-encoding to TS1/3 on input line 328. +Package textcomp Info: Setting hls sub-encoding to TS1/3 on input line 329. +Package textcomp Info: Setting hlst sub-encoding to TS1/3 on input line 330. +Package textcomp Info: Setting hlct sub-encoding to TS1/5 on input line 331. +Package textcomp Info: Setting hlx sub-encoding to TS1/5 on input line 332. +Package textcomp Info: Setting hlce sub-encoding to TS1/5 on input line 333. +Package textcomp Info: Setting hlcn sub-encoding to TS1/5 on input line 334. +Package textcomp Info: Setting hlcw sub-encoding to TS1/5 on input line 335. +Package textcomp Info: Setting hlcf sub-encoding to TS1/5 on input line 336. +Package textcomp Info: Setting pplx sub-encoding to TS1/3 on input line 337. +Package textcomp Info: Setting pplj sub-encoding to TS1/3 on input line 338. +Package textcomp Info: Setting ptmx sub-encoding to TS1/4 on input line 339. +Package textcomp Info: Setting ptmj sub-encoding to TS1/4 on input line 340. +) +(/usr/share/texlive/texmf-dist/tex/latex/listings/lstlang1.sty +File: lstlang1.sty 2013/08/26 1.5b listings language file +) +(/usr/share/texlive/texmf-dist/tex/latex/listings/lstlang1.sty +File: lstlang1.sty 2013/08/26 1.5b listings language file +) +(/usr/share/texlive/texmf-dist/tex/latex/listings/lstmisc.sty +File: lstmisc.sty 2013/08/26 1.5b (Carsten Heinz) +) + +LaTeX Warning: Unused global option(s): + [nocopyrightspace]. + +(./paper.aux) +\openout1 = `paper.aux'. + +LaTeX Font Info: Checking defaults for OML/cmm/m/it on input line 150. +LaTeX Font Info: ... okay on input line 150. +LaTeX Font Info: Checking defaults for T1/cmr/m/n on input line 150. +LaTeX Font Info: ... okay on input line 150. +LaTeX Font Info: Checking defaults for OT1/cmr/m/n on input line 150. +LaTeX Font Info: ... okay on input line 150. +LaTeX Font Info: Checking defaults for OMS/cmsy/m/n on input line 150. +LaTeX Font Info: ... okay on input line 150. +LaTeX Font Info: Checking defaults for OMX/cmex/m/n on input line 150. +LaTeX Font Info: ... okay on input line 150. +LaTeX Font Info: Checking defaults for U/cmr/m/n on input line 150. +LaTeX Font Info: ... okay on input line 150. +LaTeX Font Info: Checking defaults for PD1/pdf/m/n on input line 150. +LaTeX Font Info: ... okay on input line 150. +LaTeX Font Info: Checking defaults for TS1/cmr/m/n on input line 150. +LaTeX Font Info: Try loading font information for TS1+cmr on input line 150. + + (/usr/share/texlive/texmf-dist/tex/latex/base/ts1cmr.fd +File: ts1cmr.fd 1999/05/25 v2.5h Standard LaTeX font definitions +) +LaTeX Font Info: ... okay on input line 150. +LaTeX Font Info: Try loading font information for T1+ptm on input line 150. + +(/usr/share/texlive/texmf-dist/tex/latex/psnfss/t1ptm.fd +File: t1ptm.fd 2001/06/04 font definitions for T1/ptm. +) +(/usr/share/texlive/texmf-dist/tex/context/base/supp-pdf.mkii +[Loading MPS to PDF converter (version 2006.09.02).] +\scratchcounter=\count137 +\scratchdimen=\dimen132 +\scratchbox=\box36 +\nofMPsegments=\count138 +\nofMParguments=\count139 +\everyMPshowfont=\toks42 +\MPscratchCnt=\count140 +\MPscratchDim=\dimen133 +\MPnumerator=\count141 +\makeMPintoPDFobject=\count142 +\everyMPtoPDFconversion=\toks43 +) +\c@lstlisting=\count143 +\AtBeginShipoutBox=\box37 + + +Package hyperref Warning: Height of page (\paperheight) is invalid (0.0pt), +(hyperref) using 11in. + + +Package hyperref Warning: Draft mode on. + +Package hyperref Info: Link coloring ON on input line 150. +(/usr/share/texlive/texmf-dist/tex/latex/hyperref/nameref.sty +Package: nameref 2012/10/27 v2.43 Cross-referencing by name of section + +(/usr/share/texlive/texmf-dist/tex/generic/oberdiek/gettitlestring.sty +Package: gettitlestring 2010/12/03 v1.4 Cleanup title references (HO) +) +\c@section@level=\count144 +) +LaTeX Info: Redefining \ref on input line 150. +LaTeX Info: Redefining \pageref on input line 150. +LaTeX Info: Redefining \nameref on input line 150. +LaTeX Font Info: Try loading font information for U+msa on input line 162. + +(/usr/share/texlive/texmf-dist/tex/latex/amsfonts/umsa.fd +File: umsa.fd 2013/01/14 v3.01 AMS symbols A +) +LaTeX Font Info: Try loading font information for U+msb on input line 162. + +(/usr/share/texlive/texmf-dist/tex/latex/amsfonts/umsb.fd +File: umsb.fd 2013/01/14 v3.01 AMS symbols B +) +LaTeX Font Info: Try loading font information for U+lasy on input line 162. + +(/usr/share/texlive/texmf-dist/tex/latex/base/ulasy.fd +File: ulasy.fd 1998/08/17 v2.2e LaTeX symbol font definitions +) (./abstract.tex) +(./introduction.tex +Underfull \hbox (badness 1622) in paragraph at lines 15--26 +[]\T1/ptm/m/n/9 The C/C++ stan-dard com-mit-tee ex-tended the C and C++ + [] + + +Underfull \hbox (badness 1062) in paragraph at lines 27--38 +[]\T1/ptm/m/n/9 Researchers have de-vel-oped tools for ex-plor-ing the be-hav- + [] + + +Underfull \vbox (badness 7116) has occurred while \output is active [] + + [1{/var/lib/texmf/fonts/map/pdftex/updmap/pdftex.map} + + +] +LaTeX Font Info: Font shape `T1/ptm/bx/n' in size <9> not available +(Font) Font shape `T1/ptm/b/n' tried instead on input line 76. +LaTeX Font Info: Try loading font information for T1+pcr on input line 119. + +(/usr/share/texlive/texmf-dist/tex/latex/psnfss/t1pcr.fd +File: t1pcr.fd 2001/06/04 font definitions for T1/pcr. +) +Underfull \hbox (badness 10000) in paragraph at lines 142--163 +[]\T1/ptm/b/n/9 Constraining Re-order-ings (Spec-i-fy-ing Syn-chro-niza-tion + [] + + +File: figures/specworkflow.pdf Graphic file (type pdf) + + +Package pdftex.def Info: figures/specworkflow.pdf used on input line 174. +(pdftex.def) Requested size: 149.31004pt x 113.82695pt. +LaTeX Font Info: Font shape `T1/ptm/bx/sc' in size <9> not available +(Font) Font shape `T1/ptm/b/sc' tried instead on input line 176. + [2 <./figures/specworkflow.pdf>]) +(./formalization.tex +Underfull \hbox (badness 4981) in paragraph at lines 34--40 +\T1/ptm/b/n/9 Hoisting loads: $\OMS/cmsy/m/n/9 8\OML/cmm/m/it/9 X; Y; X \OMS/cm +sy/m/n/9 2 [] ^ \OML/cmm/m/it/9 Y \OMS/cmsy/m/n/9 2 + [] + +) (./paper.bbl [3]) +Package atveryend Info: Empty hook `BeforeClearDocument' on input line 182. + [4 + +] +Package atveryend Info: Empty hook `AfterLastShipout' on input line 182. + (./paper.aux) +Package atveryend Info: Empty hook `AtVeryEndDocument' on input line 182. +Package atveryend Info: Empty hook `AtEndAfterFileList' on input line 182. +Package atveryend Info: Empty hook `AtVeryVeryEnd' on input line 182. + ) +Here is how much of TeX's memory you used: + 9020 strings out of 493308 + 129583 string characters out of 6140092 + 252583 words of memory out of 5000000 + 12319 multiletter control sequences out of 15000+600000 + 67735 words of font info for 70 fonts, out of 8000000 for 9000 + 957 hyphenation exceptions out of 8191 + 41i,12n,58p,446b,418s stack positions out of 5000i,500n,10000p,200000b,80000s +{/usr/share/texlive/texmf-dist/fonts/enc/dvips/base/8r.enc} +Output written on paper.pdf (4 pages, 147616 bytes). +PDF statistics: + 64 PDF objects out of 1000 (max. 8388607) + 0 named destinations out of 1000 (max. 500000) + 6 words of extra memory for PDF output out of 10000 (max. 10000000) + diff --git a/correctness-model/writeup/paper.pdf b/correctness-model/writeup/paper.pdf new file mode 100644 index 0000000..e32ddb8 Binary files /dev/null and b/correctness-model/writeup/paper.pdf differ diff --git a/correctness-model/writeup/paper.tex b/correctness-model/writeup/paper.tex new file mode 100644 index 0000000..d65f512 --- /dev/null +++ b/correctness-model/writeup/paper.tex @@ -0,0 +1,182 @@ +%----------------------------------------------------------------------------- +% +% Template for sigplanconf LaTeX Class +% +% Name: sigplanconf-template.tex +% +% Purpose: A template for sigplanconf.cls, which is a LaTeX 2e class +% file for SIGPLAN conference proceedings. +% +% Guide: Refer to "Author's Guide to the ACM SIGPLAN Class," +% sigplanconf-guide.pdf +% +% Author: Paul C. Anagnostopoulos +% Windfall Software +% 978 371-2316 +% paul@windfall.com +% +% Created: 15 February 2005 +% +%----------------------------------------------------------------------------- + + +\documentclass[nocopyrightspace]{sig-alternate} + +% The following \documentclass options may be useful: +% +% 10pt To set in 10-point type instead of 9-point. +% 11pt To set in 11-point type instead of 9-point. +% authoryear To obtain author/year citation style instead of numeric. +\usepackage{times} +\usepackage[scaled]{helvet} % see www.ctan.org/get/macros/latex/required/psnfss/psnfss2e.pdf +\usepackage{url} % format URLs +\usepackage{listings} % format code +\usepackage{enumitem} % adjust spacing in enums +\sloppy +\usepackage[colorlinks=true,allcolors=blue,breaklinks,draft]{hyperref} +\newcommand{\doi}[1]{doi:~\href{http://dx.doi.org/#1}{\Hurl{#1}}} % print a hyperlinked DOI + +\usepackage{rotating} +\usepackage{algorithm}% http://ctan.org/pkg/algorithms +\usepackage{algpseudocode}% http://ctan.org/pkg/algorithmicx +%\usepackage{syntax} +%\usepackage{times} + +\usepackage{amsmath} +\usepackage{graphicx} +\usepackage{xspace} +\usepackage{amssymb} + +\usepackage{listings} +\usepackage{color} +\usepackage{textcomp} +\definecolor{listinggray}{gray}{0.9} +\definecolor{lbcolor}{rgb}{0.9,0.9,0.9} +\lstset{ + % backgroundcolor=\color{lbcolor}, + % rulecolor=, + % aboveskip={1.5\baselineskip}, + % extendedchars=true, + tabsize=3, + language=C++, + basicstyle=\ttfamily\scriptsize, + upquote=true, + columns=fixed, + showstringspaces=false, + breaklines=true, + prebreak = \raisebox{0ex}[0ex][0ex]{\ensuremath{\hookleftarrow}}, + % frame=single, + showtabs=false, + showspaces=false, + showstringspaces=false, + identifierstyle=\ttfamily, + keywordstyle=\color[rgb]{0,0,1}, + commentstyle=\color[rgb]{0.133,0.545,0.133}, + stringstyle=\color[rgb]{0.627,0.126,0.941}, + numbers=left, + numberstyle=\tiny, + numbersep=8pt, + escapeinside={/*@}{@*/} +} + +\usepackage{algpseudocode} + +%% Not used yet +% \usepackage{times} +% \usepackage{latexsym} +% \usepackage{amsfonts} +% \usepackage{amsthm} +% \usepackage{txfonts} +% \usepackage{url} +% \usepackage{subfigure} + +% \usepackage{bbm} +% \usepackage{stfloats} + +\newcommand{\squishlist}{ + \begin{list}{$\bullet$} + { + \setlength{\itemsep}{1pt} + \setlength{\parsep}{1pt} + \setlength{\topsep}{1pt} + \setlength{\partopsep}{1pt} + \setlength{\leftmargin}{1em} + \setlength{\labelwidth}{0.5em} + \setlength{\labelsep}{0.5em} + } +} + +\newcommand{\squishend}{ + \end{list} +} + +\newcommand{\squishcount}{ +% \begin{enumerate} +%} + \begin{list}{\bf \arabic{enumi}.} + { \usecounter{enumi} + \setlength{\itemsep}{1pt} \setlength{\parsep}{0pt} + \setlength{\topsep}{0pt} \setlength{\partopsep}{0pt} + \setlength{\leftmargin}{1em} \setlength{\labelwidth}{0.5em} + \setlength{\labelsep}{0.5em} } } + +\newcommand{\countend}{ +% \end{enumerate} +%} +\end{list}} + + +\newcommand{\code}[1]{\text{\tt #1}} +\newcommand{\mypara}[1]{\noindent {#1}} +\newcommand{\etal}{\textit{et al}.\xspace} +\newcommand{\cdschecker}[0]{\textsc{CDSChecker}\xspace} +\newcommand{\TOOL}[0]{\textsc{CDSSpec}\xspace} + +\newcommand{\TODO}[0]{\textbf{TODO}\xspace} +\newcommand{\todo}[1]{{\bf [[#1]]}} +%\newcommand{\todo}[1]{} +\newcommand{\comment}[1]{} +\newcommand{\tuple}[1]{\ensuremath \langle #1 \rangle} +\newcommand{\rf}{\reltext{rf}} +\newcommand{\relation}[1]{\xrightarrow{\textit{#1}}} +\newcommand{\reltext}[1]{\textit{#1}} + +%\newcommand{\mysection}[1]{\vspace{-.15cm}\section{#1}\vspace{-.15cm}} +\newcommand{\mysection}[1]{\section{#1}} +%\newcommand{\mysubsection}[1]{\vspace{-.15cm}\subsection{#1}\vspace{-.1cm}} +\newcommand{\mysubsection}[1]{\subsection{#1}} +%\newcommand{\mysubsubsection}[1]{\vspace{-.15cm}\subsubsection{#1}\vspace{-.1cm}} +\newcommand{\mysubsubsection}[1]{\subsubsection{#1}} +\begin{document} + + +\sloppy + +\title{CDSSpec: Testing Concurrent Data Structures Under the C/C++11 Memory Model} + +\author{Peizhao Ou and Brian Demsky} + + + +\maketitle + +\begin{abstract} +\input{abstract} +\end{abstract} + +\input{introduction} +%\input{memorymodel} +%\input{example} +\input{formalization} +%\input{specification} +%\input{implementation} +%\input{evaluation} +%\input{related} +%\input{conclusion} + + +% We recommend abbrvnat bibliography style. +\bibliographystyle{abbrv} +\bibliography{confstrs-long,paper} + +\end{document} diff --git a/correctness-model/writeup/related.tex b/correctness-model/writeup/related.tex new file mode 100644 index 0000000..b8f025d --- /dev/null +++ b/correctness-model/writeup/related.tex @@ -0,0 +1,133 @@ +\mysection{\label{sec:related}Related Work} + +% Related work should be related to bug finding on concurrent data structures: +% 1. Specification: +% CONCURRIT: A Domain Specific Language for Reproducing Concurrency Bugs (DSL +% for reproducing bugs, allowing specifying a set of thread schedules to explore +% in order to exhibit the bug) (PLDI'13) + +% (specify the correctness of concurrent programs using +% nondeterministic sequential specifications, which allows the functional +% correctness to be understood and verified on a sequential version of the +% program) (PLDI'11) + +% (this is the extended work of NDSeq, which is +% inferring the specification) (ppopp'12) + +% (specification, not exhaustive approach, dynamic checking) +% (PLDI'05) + + +% 2. Linearizability: +% (a more complete technique +% for verifying atomicity using SPIN model checker)(SPIN Workshop'04) + +% (theoretical paper about proving a lower +% level spscification correctly implements a higher level one) (Theoretical +% Computer Science'91) + +% (static analysis +% for verifying linearizability of concurrent unbounded linked data +% structures) (CAV'07) + +% (a formal +% verification of LazyList, concurrent list-based set algorithm, shows it's +% linearizable) (CAV'06) + +% (no specification, +% run a sequential version in parallel and checks if the concurrent execution +% violates linearizability, rely on CHESS, which assumes SC) (PLDI'10) + +% (using abstract +% interpretation, shape analysis for concurrent data structures, can prove +% linearizability in a limited way) (CAV'08) + +% (SC-based checking +% linearizability) (SPIN Workshop'09) + +% (heap-allocated data +% structures, shape analysis, assuming SC) (VMCAI'09) + +% 3. General concurrency testing +% (a prioritized +% search technique that combines the speed benefit of heuristic-guided fuzzing +% with the soundness, progress, and reproducibility of stateless model checking) +% (ppopp'10) + +% (one of the early paper) (Journal of Parallel and +% Distributed Computing) + +% 4. For relaxed M.M. +% () (PLDI'07) (TODO: Should read this in more detail) + +% (a tool that test +% concurrent programs under relaxed M.M., but no specification, and not +% an exhaustive approach) (issta'11) + +Researchers have proposed and designed specifications and approaches +to find bugs in concurrent data structures based on linearization. +Early work by Wing and +Gong~\cite{test-verify-concurrent-objects} proposed using +linearizability to test and verify concurrent objects. +Line-up~\cite{lineup} builds on the Chess~\cite{chess} model checker +to automatically check deterministic linearizability. It +automatically generates the sequential specification by systematically +enumerating all sequential behaviors. +Paraglider~\cite{VechevMCLinear} supports checking with and without +linearization points based on SPIN~\cite{spinmodel}. All of these +approaches assume that there exist a sequential history that is consistent with program order. Furthermore, they also assume +the SC memory model and a trace that provides an +ordering for method invocation and response events. Our work extends the notion of equivalence to sequential executions to the relaxed memory models used +by real systems. + +Amit \etal~\cite{abstraction-linearizability} present a static +analysis based on TVLA for verifying linearizability of concurrent +linked data structures. Valeiadis~\cite{shape-value-abstraction} +demonstrates a shape-value abstraction which can automatically prove +linearizability. Thread quantification can also verify +data structure linearizability~\cite{thread-quantification}. +Colvin \etal formally verified a list-based set~\cite{formal-verification-set}. While these approaches +provide stronger guarantees than \TOOL, they were typically used to +check simpler data structures and require experts to use. +Moreover, they target the SC memory model. + +Researchers have proposed specification languages for concurrent +data structures. Refinement mapping~\cite{refinement-mapping} +provides the theoretical basis for designing and using specifications. +Commit atomicity~\cite{spin-commit-atomicity} can verify atomicity +properties. Concurrit~\cite{concurrit} is a domain-specific language +that allows programmers to write scripts to specify thread schedules +to reproduce bugs, and is useful when programmers already have some +knowledge about a bug. NDetermin~\cite{ndetermin} infers +nondeterministic sequential specifications to model the behaviors of +parallel code. + +VYRD~\cite{vyrd} is conceptually similar to \TOOL --- developers +specify commit points for concurrent code. The parallel code is then +executed and the commit points are used to identify a sequential +execution that should have the same behaviors. VYRD was designed for +the SC memory model --- it +is unable to construct a sequential refinement for a relaxed memory +model or check synchronization properties. + +{\sc Gambit}~\cite{gambit} uses a prioritized search technique that +combines stateless model checking and heuristic-guided fuzzing to unit +test code under the SC memory model. {\sc Relaxed}~\cite{memtest} +explores SC executions to identify executions with races and then +re-executes the program under the PSO or TSO memory model to test +whether the relaxations expose bugs. CheckFence~\cite{checkfence} is +a tool for verifying data structures against relaxed memory models and +takes a SAT-based approach instead of the stateless model checking +approach used by \cdschecker. + +Researchers have developed verification techniques for code that +admits only SC executions under the TSO and PSO memory +models~\cite{koushiksc,burckhardtverif}. The basic idea is to develop +an execution monitor that can detect whether non-SC executions exist +by examining only SC executions. + + diff --git a/correctness-model/writeup/sig-alternate.cls b/correctness-model/writeup/sig-alternate.cls new file mode 100644 index 0000000..694a897 --- /dev/null +++ b/correctness-model/writeup/sig-alternate.cls @@ -0,0 +1,1649 @@ +% SIG-ALTERNATE.CLS - VERSION 2.5 +% "COMPATIBLE" WITH THE "ACM_PROC_ARTICLE-SP.CLS" V3.2SP +% Gerald Murray - May 23rd 2012 +% +% ---- Start of 'updates' ---- +% Changed $10 fee to $15 -- May 2012 -- Gerry +% Changed $5 fee to $10 -- April 2009 -- Gerry +% April 22nd. 2009 - Fixed 'Natbib' incompatibility problem - Gerry +% April 22nd. 2009 - Fixed 'Babel' incompatibility problem - Gerry +% April 22nd. 2009 - Inserted various bug-fixes and improvements - Gerry +% +% To produce Type 1 fonts in the document plus allow for 'normal LaTeX accenting' in the critical areas; +% title, author block, section-heads, confname, etc. etc. +% i.e. the whole purpose of this version update is to NOT resort to 'inelegant accent patches'. +% After much research, three extra .sty packages were added to the the tail (ae, aecompl, aeguill) to solve, +% in particular, the accenting problem(s). We _could_ ask authors (via instructions/sample file) to 'include' these in +% the source .tex file - in the preamble - but if everything is already provided ('behind the scenes' - embedded IN the .cls) +% then this is less work for authors and also makes everything appear 'vanilla'. +% NOTE: all 'patchwork accenting" has been commented out (here) and is no longer 'used' in the sample .tex file (either). +% Gerry June 2007 +% +% Patch for accenting in conference name/location. Gerry May 3rd. 2007 +% Rule widths changed to .5, author count (>6) fixed, roll-back for Type 3 problem. Gerry March 20th. 2007 +% Changes made to 'modernize' the fontnames but esp. for MikTeX users V2.4/2.5 - Nov. 30th. 2006 +% Updated the \email definition to allow for its use inside of 'shared affiliations' - Nov. 30th. 2006 +% Fixed the 'section number depth value' - Nov. 30th. 2006 +% +% Footnotes inside table cells using \minipage (Oct. 2002) +% Georgia fixed bug in sub-sub-section numbering in paragraphs (July 29th. 2002) +% JS/GM fix to vertical spacing before Proofs (July 30th. 2002) +% +% Made the Permission Statement / Conference Info / Copyright Info +% 'user definable' in the source .tex file OR automatic if +% not specified. +% +% Allowance made to switch default fonts between those systems using +% normal/modern font names and those using 'Type 1' or 'Truetype' fonts. +% See LINE NUMBER 255 for details. +% Also provided for enumerated/annotated Corollaries 'surrounded' by +% enumerated Theorems (line 848). +% Gerry November 11th. 1999 +% +% ---- End of 'updates' ---- +% +\def\fileversion{v2.5} % for ACM's tracking purposes +\def\filedate{May 23, 2012} % Gerry Murray's tracking data +\def\docdate {Wednesday 23rd. May 2012} % Gerry Murray (with deltas to doc} +\usepackage{epsfig} +\usepackage{amssymb} +\usepackage{amsmath} +\usepackage{amsfonts} +% Need this for accents in Arial/Helvetica +%\usepackage[T1]{fontenc} % Gerry March 12, 2007 - causes Type 3 problems (body text) +%\usepackage{textcomp} +% +% SIG-ALTERNATE DOCUMENT STYLE +% G.K.M. Tobin August-October 1999 +% adapted from ARTICLE document style by Ken Traub, Olin Shivers +% also using elements of esub2acm.cls +% HEAVILY MODIFIED, SUBSEQUENTLY, BY GERRY MURRAY 2000 +% ARTICLE DOCUMENT STYLE -- Released 16 March 1988 +% for LaTeX version 2.09 +% Copyright (C) 1988 by Leslie Lamport +% +% +%%% sig-alternate.cls is an 'ALTERNATE' document style for producing +%%% two-column camera-ready pages for ACM conferences. +%%% THIS FILE DOES NOT STRICTLY ADHERE TO THE SIGS (BOARD-ENDORSED) +%%% PROCEEDINGS STYLE. It has been designed to produce a 'tighter' +%%% paper in response to concerns over page budgets. +%%% The main features of this style are: +%%% +%%% 1) Two columns. +%%% 2) Side and top margins of 4.5pc, bottom margin of 6pc, column gutter of +%%% 2pc, hence columns are 20pc wide and 55.5pc tall. (6pc =3D 1in, approx) +%%% 3) First page has title information, and an extra 6pc of space at the +%%% bottom of the first column for the ACM copyright notice. +%%% 4) Text is 9pt on 10pt baselines; titles (except main) are 9pt bold. +%%% +%%% +%%% There are a few restrictions you must observe: +%%% +%%% 1) You cannot change the font size; ACM wants you to use 9pt. +%%% 3) You must start your paper with the \maketitle command. Prior to the +%%% \maketitle you must have \title and \author commands. If you have a +%%% \date command it will be ignored; no date appears on the paper, since +%%% the proceedings will have a date on the front cover. +%%% 4) Marginal paragraphs, tables of contents, lists of figures and tables, +%%% and page headings are all forbidden. +%%% 5) The `figure' environment will produce a figure one column wide; if you +%%% want one that is two columns wide, use `figure*'. +%%% +% +%%% Copyright Space: +%%% This style automatically reserves 1" blank space at the bottom of page 1/ +%%% column 1. This space can optionally be filled with some text using the +%%% \toappear{...} command. If used, this command must be BEFORE the \maketitle +%%% command. If this command is defined AND [preprint] is on, then the +%%% space is filled with the {...} text (at the bottom); otherwise, it is +%%% blank. If you use \toappearbox{...} instead of \toappear{...} then a +%%% box will be drawn around the text (if [preprint] is on). +%%% +%%% A typical usage looks like this: +%%% \toappear{To appear in the Ninth AES Conference on Medievil Lithuanian +%%% Embalming Technique, June 1991, Alfaretta, Georgia.} +%%% This will be included in the preprint, and left out of the conference +%%% version. +%%% +%%% WARNING: +%%% Some dvi-ps converters heuristically allow chars to drift from their +%%% true positions a few pixels. This may be noticeable with the 9pt sans-serif +%%% bold font used for section headers. +%%% You may turn this hackery off via the -e option: +%%% dvips -e 0 foo.dvi >foo.ps +%%% +\typeout{Document Class 'sig-alternate' <23rd. May '12>. Modified by G.K.M. Tobin/Gerry Murray} +\typeout{Based in part upon document Style `acmconf' <22 May 89>. Hacked 4/91 by} +\typeout{shivers@cs.cmu.edu, 4/93 by theobald@cs.mcgill.ca} +\typeout{Excerpts were taken from (Journal Style) 'esub2acm.cls'.} +\typeout{****** Bugs/comments/suggestions/technicalities to Gerry Murray -- murray@hq.acm.org ******} +\typeout{Questions on the style, SIGS policies, etc. to Adrienne Griscti griscti@acm.org} +\oddsidemargin 4.5pc +\evensidemargin 4.5pc +\advance\oddsidemargin by -1in % Correct for LaTeX gratuitousness +\advance\evensidemargin by -1in % Correct for LaTeX gratuitousness +\marginparwidth 0pt % Margin pars are not allowed. +\marginparsep 11pt % Horizontal space between outer margin and + % marginal note + + % Top of page: +\topmargin 4.5pc % Nominal distance from top of page to top of + % box containing running head. +\advance\topmargin by -1in % Correct for LaTeX gratuitousness +\headheight 0pt % Height of box containing running head. +\headsep 0pt % Space between running head and text. + % Bottom of page: +\footskip 30pt % Distance from baseline of box containing foot + % to baseline of last line of text. +\@ifundefined{footheight}{\newdimen\footheight}{}% this is for LaTeX2e +\footheight 12pt % Height of box containing running foot. + +%% Must redefine the top margin so there's room for headers and +%% page numbers if you are using the preprint option. Footers +%% are OK as is. Olin. +\advance\topmargin by -37pt % Leave 37pt above text for headers +\headheight 12pt % Height of box containing running head. +\headsep 25pt % Space between running head and text. + +\textheight 666pt % 9 1/4 column height +\textwidth 42pc % Width of text line. + % For two-column mode: +\columnsep 2pc % Space between columns +\columnseprule 0pt % Width of rule between columns. +\hfuzz 1pt % Allow some variation in column width, otherwise it's + % too hard to typeset in narrow columns. + +\footnotesep 5.6pt % Height of strut placed at the beginning of every + % footnote =3D height of normal \footnotesize strut, + % so no extra space between footnotes. + +\skip\footins 8.1pt plus 4pt minus 2pt % Space between last line of text and + % top of first footnote. +\floatsep 11pt plus 2pt minus 2pt % Space between adjacent floats moved + % to top or bottom of text page. +\textfloatsep 18pt plus 2pt minus 4pt % Space between main text and floats + % at top or bottom of page. +\intextsep 11pt plus 2pt minus 2pt % Space between in-text figures and + % text. +\@ifundefined{@maxsep}{\newdimen\@maxsep}{}% this is for LaTeX2e +\@maxsep 18pt % The maximum of \floatsep, + % \textfloatsep and \intextsep (minus + % the stretch and shrink). +\dblfloatsep 11pt plus 2pt minus 2pt % Same as \floatsep for double-column + % figures in two-column mode. +\dbltextfloatsep 18pt plus 2pt minus 4pt% \textfloatsep for double-column + % floats. +\@ifundefined{@dblmaxsep}{\newdimen\@dblmaxsep}{}% this is for LaTeX2e +\@dblmaxsep 18pt % The maximum of \dblfloatsep and + % \dbltexfloatsep. +\@fptop 0pt plus 1fil % Stretch at top of float page/column. (Must be + % 0pt plus ...) +\@fpsep 8pt plus 2fil % Space between floats on float page/column. +\@fpbot 0pt plus 1fil % Stretch at bottom of float page/column. (Must be + % 0pt plus ... ) +\@dblfptop 0pt plus 1fil % Stretch at top of float page. (Must be 0pt plus ...) +\@dblfpsep 8pt plus 2fil % Space between floats on float page. +\@dblfpbot 0pt plus 1fil % Stretch at bottom of float page. (Must be + % 0pt plus ... ) +\marginparpush 5pt % Minimum vertical separation between two marginal + % notes. + +\parskip 0pt plus 1pt % Extra vertical space between paragraphs. +\parindent 9pt % GM July 2000 / was 0pt - width of paragraph indentation. +\partopsep 2pt plus 1pt minus 1pt% Extra vertical space, in addition to + % \parskip and \topsep, added when user + % leaves blank line before environment. + +\@lowpenalty 51 % Produced by \nopagebreak[1] or \nolinebreak[1] +\@medpenalty 151 % Produced by \nopagebreak[2] or \nolinebreak[2] +\@highpenalty 301 % Produced by \nopagebreak[3] or \nolinebreak[3] + +\@beginparpenalty -\@lowpenalty % Before a list or paragraph environment. +\@endparpenalty -\@lowpenalty % After a list or paragraph environment. +\@itempenalty -\@lowpenalty % Between list items. + +%\@namedef{ds@10pt}{\@latexerr{The `10pt' option is not allowed in the `acmconf' +\@namedef{ds@10pt}{\ClassError{The `10pt' option is not allowed in the `acmconf' % January 2008 + document style.}\@eha} +%\@namedef{ds@11pt}{\@latexerr{The `11pt' option is not allowed in the `acmconf' +\@namedef{ds@11pt}{\ClassError{The `11pt' option is not allowed in the `acmconf' % January 2008 + document style.}\@eha} +%\@namedef{ds@12pt}{\@latexerr{The `12pt' option is not allowed in the `acmconf' +\@namedef{ds@12pt}{\ClassError{The `12pt' option is not allowed in the `acmconf' % January 2008 + document style.}\@eha} + +\@options + +\lineskip 2pt % \lineskip is 1pt for all font sizes. +\normallineskip 2pt +\def\baselinestretch{1} + +\abovedisplayskip 9pt plus2pt minus4.5pt% +\belowdisplayskip \abovedisplayskip +\abovedisplayshortskip \z@ plus3pt% +\belowdisplayshortskip 5.4pt plus3pt minus3pt% +\let\@listi\@listI % Setting of \@listi added 9 Jun 87 + +\def\small{\@setsize\small{9pt}\viiipt\@viiipt +\abovedisplayskip 7.6pt plus 3pt minus 4pt% +\belowdisplayskip \abovedisplayskip +\abovedisplayshortskip \z@ plus2pt% +\belowdisplayshortskip 3.6pt plus2pt minus 2pt +\def\@listi{\leftmargin\leftmargini %% Added 22 Dec 87 +\topsep 4pt plus 2pt minus 2pt\parsep 2pt plus 1pt minus 1pt +\itemsep \parsep}} + +\def\footnotesize{\@setsize\footnotesize{9pt}\ixpt\@ixpt +\abovedisplayskip 6.4pt plus 2pt minus 4pt% +\belowdisplayskip \abovedisplayskip +\abovedisplayshortskip \z@ plus 1pt% +\belowdisplayshortskip 2.7pt plus 1pt minus 2pt +\def\@listi{\leftmargin\leftmargini %% Added 22 Dec 87 +\topsep 3pt plus 1pt minus 1pt\parsep 2pt plus 1pt minus 1pt +\itemsep \parsep}} + +\newcount\aucount +\newcount\originalaucount +\newdimen\auwidth +\auwidth=\textwidth +\newdimen\auskip +\newcount\auskipcount +\newdimen\auskip +\global\auskip=1pc +\newdimen\allauboxes +\allauboxes=\auwidth +\newtoks\addauthors +\newcount\addauflag +\global\addauflag=0 %Haven't shown additional authors yet + +\newtoks\subtitletext +\gdef\subtitle#1{\subtitletext={#1}} + +\gdef\additionalauthors#1{\addauthors={#1}} + +\gdef\numberofauthors#1{\global\aucount=#1 +\ifnum\aucount>3\global\originalaucount=\aucount \global\aucount=3\fi %g} % 3 OK - Gerry March 2007 +\global\auskipcount=\aucount\global\advance\auskipcount by 1 +\global\multiply\auskipcount by 2 +\global\multiply\auskip by \auskipcount +\global\advance\auwidth by -\auskip +\global\divide\auwidth by \aucount} + +% \and was modified to count the number of authors. GKMT 12 Aug 1999 +\def\alignauthor{% % \begin{tabular} +\end{tabular}% + \begin{tabular}[t]{p{\auwidth}}\centering}% + +% *** NOTE *** NOTE *** NOTE *** NOTE *** +% If you have 'font problems' then you may need +% to change these, e.g. 'arialb' instead of "arialbd". +% Gerry Murray 11/11/1999 +% *** OR ** comment out block A and activate block B or vice versa. +% ********************************************** +% +% -- Start of block A -- (Type 1 or Truetype fonts) +%\newfont{\secfnt}{timesbd at 12pt} % was timenrb originally - now is timesbd +%\newfont{\secit}{timesbi at 12pt} %13 Jan 00 gkmt +%\newfont{\subsecfnt}{timesi at 11pt} % was timenrri originally - now is timesi +%\newfont{\subsecit}{timesbi at 11pt} % 13 Jan 00 gkmt -- was times changed to timesbi gm 2/4/2000 +% % because "normal" is italic, "italic" is Roman +%\newfont{\ttlfnt}{arialbd at 18pt} % was arialb originally - now is arialbd +%\newfont{\ttlit}{arialbi at 18pt} % 13 Jan 00 gkmt +%\newfont{\subttlfnt}{arial at 14pt} % was arialr originally - now is arial +%\newfont{\subttlit}{ariali at 14pt} % 13 Jan 00 gkmt +%\newfont{\subttlbf}{arialbd at 14pt} % 13 Jan 00 gkmt +%\newfont{\aufnt}{arial at 12pt} % was arialr originally - now is arial +%\newfont{\auit}{ariali at 12pt} % 13 Jan 00 gkmt +%\newfont{\affaddr}{arial at 10pt} % was arialr originally - now is arial +%\newfont{\affaddrit}{ariali at 10pt} %13 Jan 00 gkmt +%\newfont{\eaddfnt}{arial at 12pt} % was arialr originally - now is arial +%\newfont{\ixpt}{times at 9pt} % was timenrr originally - now is times +%\newfont{\confname}{timesi at 8pt} % was timenrri - now is timesi +%\newfont{\crnotice}{times at 8pt} % was timenrr originally - now is times +%\newfont{\ninept}{times at 9pt} % was timenrr originally - now is times + +% ********************************************* +% -- End of block A -- +% +% +% -- Start of block B -- UPDATED FONT NAMES +% ********************************************* +% Gerry Murray 11/30/2006 +% ********************************************* +\newfont{\secfnt}{ptmb8t at 12pt} +\newfont{\secit}{ptmbi8t at 12pt} %13 Jan 00 gkmt +\newfont{\subsecfnt}{ptmri8t at 11pt} +\newfont{\subsecit}{ptmbi8t at 11pt} % +\newfont{\ttlfnt}{phvb8t at 18pt} +\newfont{\ttlit}{phvbo8t at 18pt} % GM 2/4/2000 +\newfont{\subttlfnt}{phvr8t at 14pt} +\newfont{\subttlit}{phvro8t at 14pt} % GM 2/4/2000 +\newfont{\subttlbf}{phvb8t at 14pt} % 13 Jan 00 gkmt +\newfont{\aufnt}{phvr8t at 12pt} +\newfont{\auit}{phvro8t at 12pt} % GM 2/4/2000 +\newfont{\affaddr}{phvr8t at 10pt} +\newfont{\affaddrit}{phvro8t at 10pt} % GM 2/4/2000 +\newfont{\eaddfnt}{phvr8t at 12pt} +\newfont{\ixpt}{ptmr8t at 9pt} +\newfont{\confname}{ptmri8t at 8pt} +\newfont{\crnotice}{ptmr8t at 8pt} +\newfont{\ninept}{ptmr8t at 9pt} +% +++++++++++++++++++++++++++++++++++++++++++++ +% -- End of block B -- + +%\def\email#1{{{\eaddfnt{\vskip 4pt#1}}}} +% If we have an email, inside a "shared affiliation" then we need the following instead +\def\email#1{{{\eaddfnt{\par #1}}}} % revised - GM - 11/30/2006 + +\def\addauthorsection{\ifnum\originalaucount>6 % was 3 - Gerry March 2007 + \section{Additional Authors}\the\addauthors + \fi} + +\newcount\savesection +\newcount\sectioncntr +\global\sectioncntr=1 + +\setcounter{secnumdepth}{3} + +\def\appendix{\par +\section*{APPENDIX} +\setcounter{section}{0} + \setcounter{subsection}{0} + \def\thesection{\Alph{section}} } + +\leftmargini 22.5pt +\leftmarginii 19.8pt % > \labelsep + width of '(m)' +\leftmarginiii 16.8pt % > \labelsep + width of 'vii.' +\leftmarginiv 15.3pt % > \labelsep + width of 'M.' +\leftmarginv 9pt +\leftmarginvi 9pt + +\leftmargin\leftmargini +\labelsep 4.5pt +\labelwidth\leftmargini\advance\labelwidth-\labelsep + +\def\@listI{\leftmargin\leftmargini \parsep 3.6pt plus 2pt minus 1pt% +\topsep 7.2pt plus 2pt minus 4pt% +\itemsep 3.6pt plus 2pt minus 1pt} + +\let\@listi\@listI +\@listi + +\def\@listii{\leftmargin\leftmarginii + \labelwidth\leftmarginii\advance\labelwidth-\labelsep + \topsep 3.6pt plus 2pt minus 1pt + \parsep 1.8pt plus 0.9pt minus 0.9pt + \itemsep \parsep} + +\def\@listiii{\leftmargin\leftmarginiii + \labelwidth\leftmarginiii\advance\labelwidth-\labelsep + \topsep 1.8pt plus 0.9pt minus 0.9pt + \parsep \z@ \partopsep 1pt plus 0pt minus 1pt + \itemsep \topsep} + +\def\@listiv{\leftmargin\leftmarginiv + \labelwidth\leftmarginiv\advance\labelwidth-\labelsep} + +\def\@listv{\leftmargin\leftmarginv + \labelwidth\leftmarginv\advance\labelwidth-\labelsep} + +\def\@listvi{\leftmargin\leftmarginvi + \labelwidth\leftmarginvi\advance\labelwidth-\labelsep} + +\def\labelenumi{\theenumi.} +\def\theenumi{\arabic{enumi}} + +\def\labelenumii{(\theenumii)} +\def\theenumii{\alph{enumii}} +\def\p@enumii{\theenumi} + +\def\labelenumiii{\theenumiii.} +\def\theenumiii{\roman{enumiii}} +\def\p@enumiii{\theenumi(\theenumii)} + +\def\labelenumiv{\theenumiv.} +\def\theenumiv{\Alph{enumiv}} +\def\p@enumiv{\p@enumiii\theenumiii} + +\def\labelitemi{$\bullet$} +\def\labelitemii{\bf --} +\def\labelitemiii{$\ast$} +\def\labelitemiv{$\cdot$} + +\def\verse{\let\\=\@centercr + \list{}{\itemsep\z@ \itemindent -1.5em\listparindent \itemindent + \rightmargin\leftmargin\advance\leftmargin 1.5em}\item[]} +\let\endverse\endlist + +\def\quotation{\list{}{\listparindent 1.5em + \itemindent\listparindent + \rightmargin\leftmargin \parsep 0pt plus 1pt}\item[]} +\let\endquotation=\endlist + +\def\quote{\list{}{\rightmargin\leftmargin}\item[]} +\let\endquote=\endlist + +\def\descriptionlabel#1{\hspace\labelsep \bf #1} +\def\description{\list{}{\labelwidth\z@ \itemindent-\leftmargin + \let\makelabel\descriptionlabel}} + +\let\enddescription\endlist + +\def\theequation{\arabic{equation}} + +\arraycolsep 4.5pt % Half the space between columns in an array environment. +\tabcolsep 5.4pt % Half the space between columns in a tabular environment. +\arrayrulewidth .5pt % Width of rules in array and tabular environment. % (was .4) updated Gerry March 20 2007 +\doublerulesep 1.8pt % Space between adjacent rules in array or tabular env. + +\tabbingsep \labelsep % Space used by the \' command. (See LaTeX manual.) + +\skip\@mpfootins =\skip\footins + +\fboxsep =2.7pt % Space left between box and text by \fbox and \framebox. +\fboxrule =.5pt % Width of rules in box made by \fbox and \framebox. % (was .4) updated Gerry March 20 2007 + +\def\thepart{\Roman{part}} % Roman numeral part numbers. +\def\thesection {\arabic{section}} +\def\thesubsection {\thesection.\arabic{subsection}} +%\def\thesubsubsection {\thesubsection.\arabic{subsubsection}} % GM 7/30/2002 +%\def\theparagraph {\thesubsubsection.\arabic{paragraph}} % GM 7/30/2002 +\def\thesubparagraph {\theparagraph.\arabic{subparagraph}} + +\def\@pnumwidth{1.55em} +\def\@tocrmarg {2.55em} +\def\@dotsep{4.5} +\setcounter{tocdepth}{3} + +%\def\tableofcontents{\@latexerr{\tableofcontents: Tables of contents are not +% allowed in the `acmconf' document style.}\@eha} + +\def\tableofcontents{\ClassError{% + \string\tableofcontents\space is not allowed in the `acmconf' document % January 2008 + style}\@eha} + +\def\l@part#1#2{\addpenalty{\@secpenalty} + \addvspace{2.25em plus 1pt} % space above part line + \begingroup + \@tempdima 3em % width of box holding part number, used by + \parindent \z@ \rightskip \@pnumwidth %% \numberline + \parfillskip -\@pnumwidth + {\large \bf % set line in \large boldface + \leavevmode % TeX command to enter horizontal mode. + #1\hfil \hbox to\@pnumwidth{\hss #2}}\par + \nobreak % Never break after part entry + \endgroup} + +\def\l@section#1#2{\addpenalty{\@secpenalty} % good place for page break + \addvspace{1.0em plus 1pt} % space above toc entry + \@tempdima 1.5em % width of box holding section number + \begingroup + \parindent \z@ \rightskip \@pnumwidth + \parfillskip -\@pnumwidth + \bf % Boldface. + \leavevmode % TeX command to enter horizontal mode. + \advance\leftskip\@tempdima %% added 5 Feb 88 to conform to + \hskip -\leftskip %% 25 Jan 88 change to \numberline + #1\nobreak\hfil \nobreak\hbox to\@pnumwidth{\hss #2}\par + \endgroup} + + +\def\l@subsection{\@dottedtocline{2}{1.5em}{2.3em}} +\def\l@subsubsection{\@dottedtocline{3}{3.8em}{3.2em}} +\def\l@paragraph{\@dottedtocline{4}{7.0em}{4.1em}} +\def\l@subparagraph{\@dottedtocline{5}{10em}{5em}} + +%\def\listoffigures{\@latexerr{\listoffigures: Lists of figures are not +% allowed in the `acmconf' document style.}\@eha} + +\def\listoffigures{\ClassError{% + \string\listoffigures\space is not allowed in the `acmconf' document % January 2008 + style}\@eha} + +\def\l@figure{\@dottedtocline{1}{1.5em}{2.3em}} + +%\def\listoftables{\@latexerr{\listoftables: Lists of tables are not +% allowed in the `acmconf' document style.}\@eha} +%\let\l@table\l@figure + +\def\listoftables{\ClassError{% + \string\listoftables\space is not allowed in the `acmconf' document % January 2008 + style}\@eha} + \let\l@table\l@figure + +\def\footnoterule{\kern-3\p@ + \hrule width .5\columnwidth % (was .4) updated Gerry March 20 2007 + \kern 2.6\p@} % The \hrule has default height of .4pt % (was .4) updated Gerry March 20 2007 +% ------ +\long\def\@makefntext#1{\noindent +%\hbox to .5em{\hss$^{\@thefnmark}$}#1} % original +\hbox to .5em{\hss\textsuperscript{\@thefnmark}}#1} % C. Clifton / GM Oct. 2nd. 2002 +% ------- + +\long\def\@maketntext#1{\noindent +#1} + +\long\def\@maketitlenotetext#1#2{\noindent + \hbox to 1.8em{\hss$^{#1}$}#2} + +\setcounter{topnumber}{2} +\def\topfraction{.7} +\setcounter{bottomnumber}{1} +\def\bottomfraction{.3} +\setcounter{totalnumber}{3} +\def\textfraction{.2} +\def\floatpagefraction{.5} +\setcounter{dbltopnumber}{2} +\def\dbltopfraction{.7} +\def\dblfloatpagefraction{.5} + +% +\long\def\@makecaption#1#2{ + \vskip \baselineskip + \setbox\@tempboxa\hbox{\textbf{#1: #2}} + \ifdim \wd\@tempboxa >\hsize % IF longer than one line: + \textbf{#1: #2}\par % THEN set as ordinary paragraph. + \else % ELSE center. + \hbox to\hsize{\hfil\box\@tempboxa\hfil}\par + \fi} + +% + +\long\def\@makecaption#1#2{ + \vskip 10pt + \setbox\@tempboxa\hbox{\textbf{#1: #2}} + \ifdim \wd\@tempboxa >\hsize % IF longer than one line: + \textbf{#1: #2}\par % THEN set as ordinary paragraph. + \else % ELSE center. + \hbox to\hsize{\hfil\box\@tempboxa\hfil} + \fi} + +\@ifundefined{figure}{\newcounter {figure}} % this is for LaTeX2e + +\def\fps@figure{tbp} +\def\ftype@figure{1} +\def\ext@figure{lof} +\def\fnum@figure{Figure \thefigure} +\def\figure{\@float{figure}} +%\let\endfigure\end@float +\def\endfigure{\end@float} % Gerry January 2008 +\@namedef{figure*}{\@dblfloat{figure}} +\@namedef{endfigure*}{\end@dblfloat} + +\@ifundefined{table}{\newcounter {table}} % this is for LaTeX2e + +\def\fps@table{tbp} +\def\ftype@table{2} +\def\ext@table{lot} +\def\fnum@table{Table \thetable} +\def\table{\@float{table}} +%\let\endtable\end@float +\def\endtable{\end@float} % Gerry January 2008 +\@namedef{table*}{\@dblfloat{table}} +\@namedef{endtable*}{\end@dblfloat} + +\newtoks\titleboxnotes +\newcount\titleboxnoteflag + +\def\maketitle{\par + \begingroup + \def\thefootnote{\fnsymbol{footnote}} + \def\@makefnmark{\hbox + to 0pt{$^{\@thefnmark}$\hss}} + \twocolumn[\@maketitle] +\@thanks + \endgroup + \setcounter{footnote}{0} + \let\maketitle\relax + \let\@maketitle\relax + \gdef\@thanks{}\gdef\@author{}\gdef\@title{}\gdef\@subtitle{}\let\thanks\relax + \@copyrightspace} + +%% CHANGES ON NEXT LINES +\newif\if@ll % to record which version of LaTeX is in use + +\expandafter\ifx\csname LaTeXe\endcsname\relax % LaTeX2.09 is used +\else% LaTeX2e is used, so set ll to true +\global\@lltrue +\fi + +\if@ll + \NeedsTeXFormat{LaTeX2e} + \ProvidesClass{sig-alternate} [2012/05/23 - V2.5 - based on acmproc.cls V1.3 ] + \RequirePackage{latexsym}% QUERY: are these two really needed? + \let\dooptions\ProcessOptions +\else + \let\dooptions\@options +\fi +%% END CHANGES + +\def\@height{height} +\def\@width{width} +\def\@minus{minus} +\def\@plus{plus} +\def\hb@xt@{\hbox to} +\newif\if@faircopy +\@faircopyfalse +\def\ds@faircopy{\@faircopytrue} + +\def\ds@preprint{\@faircopyfalse} + +\@twosidetrue +\@mparswitchtrue +\def\ds@draft{\overfullrule 5\p@} +%% CHANGE ON NEXT LINE +\dooptions + +\lineskip \p@ +\normallineskip \p@ +\def\baselinestretch{1} +\def\@ptsize{0} %needed for amssymbols.sty + +%% CHANGES ON NEXT LINES +\if@ll% allow use of old-style font change commands in LaTeX2e +\@maxdepth\maxdepth +% +\DeclareOldFontCommand{\rm}{\ninept\rmfamily}{\mathrm} +\DeclareOldFontCommand{\sf}{\normalfont\sffamily}{\mathsf} +\DeclareOldFontCommand{\tt}{\normalfont\ttfamily}{\mathtt} +\DeclareOldFontCommand{\bf}{\normalfont\bfseries}{\mathbf} +\DeclareOldFontCommand{\it}{\normalfont\itshape}{\mathit} +\DeclareOldFontCommand{\sl}{\normalfont\slshape}{\@nomath\sl} +\DeclareOldFontCommand{\sc}{\normalfont\scshape}{\@nomath\sc} +\DeclareRobustCommand*{\cal}{\@fontswitch{\relax}{\mathcal}} +\DeclareRobustCommand*{\mit}{\@fontswitch{\relax}{\mathnormal}} +\fi +% +\if@ll + \renewcommand{\rmdefault}{cmr} % was 'ttm' +% Note! I have also found 'mvr' to work ESPECIALLY well. +% Gerry - October 1999 +% You may need to change your LV1times.fd file so that sc is +% mapped to cmcsc - -for smallcaps -- that is if you decide +% to change {cmr} to {times} above. (Not recommended) + \renewcommand{\@ptsize}{} + \renewcommand{\normalsize}{% + \@setfontsize\normalsize\@ixpt{10.5\p@}%\ninept% + \abovedisplayskip 6\p@ \@plus2\p@ \@minus\p@ + \belowdisplayskip \abovedisplayskip + \abovedisplayshortskip 6\p@ \@minus 3\p@ + \belowdisplayshortskip 6\p@ \@minus 3\p@ + \let\@listi\@listI + } +\else + \def\@normalsize{%changed next to 9 from 10 + \@setsize\normalsize{9\p@}\ixpt\@ixpt + \abovedisplayskip 6\p@ \@plus2\p@ \@minus\p@ + \belowdisplayskip \abovedisplayskip + \abovedisplayshortskip 6\p@ \@minus 3\p@ + \belowdisplayshortskip 6\p@ \@minus 3\p@ + \let\@listi\@listI + }% +\fi +\if@ll + \newcommand\scriptsize{\@setfontsize\scriptsize\@viipt{8\p@}} + \newcommand\tiny{\@setfontsize\tiny\@vpt{6\p@}} + \newcommand\large{\@setfontsize\large\@xiipt{14\p@}} + \newcommand\Large{\@setfontsize\Large\@xivpt{18\p@}} + \newcommand\LARGE{\@setfontsize\LARGE\@xviipt{20\p@}} + \newcommand\huge{\@setfontsize\huge\@xxpt{25\p@}} + \newcommand\Huge{\@setfontsize\Huge\@xxvpt{30\p@}} +\else + \def\scriptsize{\@setsize\scriptsize{8\p@}\viipt\@viipt} + \def\tiny{\@setsize\tiny{6\p@}\vpt\@vpt} + \def\large{\@setsize\large{14\p@}\xiipt\@xiipt} + \def\Large{\@setsize\Large{18\p@}\xivpt\@xivpt} + \def\LARGE{\@setsize\LARGE{20\p@}\xviipt\@xviipt} + \def\huge{\@setsize\huge{25\p@}\xxpt\@xxpt} + \def\Huge{\@setsize\Huge{30\p@}\xxvpt\@xxvpt} +\fi +\normalsize + +% make aubox hsize/number of authors up to 3, less gutter +% then showbox gutter showbox gutter showbox -- GKMT Aug 99 +\newbox\@acmtitlebox +\def\@maketitle{\newpage + \null + \setbox\@acmtitlebox\vbox{% +\baselineskip 20pt +\vskip 2em % Vertical space above title. + \begin{center} + {\ttlfnt \@title\par} % Title set in 18pt Helvetica (Arial) bold size. + \vskip 1.5em % Vertical space after title. +%This should be the subtitle. +{\subttlfnt \the\subtitletext\par}\vskip 1.25em%\fi + {\baselineskip 16pt\aufnt % each author set in \12 pt Arial, in a + \lineskip .5em % tabular environment + \begin{tabular}[t]{c}\@author + \end{tabular}\par} + \vskip 1.5em % Vertical space after author. + \end{center}} + \dimen0=\ht\@acmtitlebox + \advance\dimen0 by -12.75pc\relax % Increased space for title box -- KBT + \unvbox\@acmtitlebox + \ifdim\dimen0<0.0pt\relax\vskip-\dimen0\fi} + + +\newcount\titlenotecount +\global\titlenotecount=0 +\newtoks\tntoks +\newtoks\tntokstwo +\newtoks\tntoksthree +\newtoks\tntoksfour +\newtoks\tntoksfive + +\def\abstract{ +\ifnum\titlenotecount>0 % was =1 + \insert\footins{% + \reset@font\footnotesize + \interlinepenalty\interfootnotelinepenalty + \splittopskip\footnotesep + \splitmaxdepth \dp\strutbox \floatingpenalty \@MM + \hsize\columnwidth \@parboxrestore + \protected@edef\@currentlabel{% + }% + \color@begingroup +\ifnum\titlenotecount=1 + \@maketntext{% + \raisebox{4pt}{$\ast$}\rule\z@\footnotesep\ignorespaces\the\tntoks\@finalstrut\strutbox}% +\fi +\ifnum\titlenotecount=2 + \@maketntext{% + \raisebox{4pt}{$\ast$}\rule\z@\footnotesep\ignorespaces\the\tntoks\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\dagger$}\rule\z@\footnotesep\ignorespaces\the\tntokstwo\@finalstrut\strutbox}% +\fi +\ifnum\titlenotecount=3 + \@maketntext{% + \raisebox{4pt}{$\ast$}\rule\z@\footnotesep\ignorespaces\the\tntoks\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\dagger$}\rule\z@\footnotesep\ignorespaces\the\tntokstwo\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\ddagger$}\rule\z@\footnotesep\ignorespaces\the\tntoksthree\@finalstrut\strutbox}% +\fi +\ifnum\titlenotecount=4 + \@maketntext{% + \raisebox{4pt}{$\ast$}\rule\z@\footnotesep\ignorespaces\the\tntoks\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\dagger$}\rule\z@\footnotesep\ignorespaces\the\tntokstwo\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\ddagger$}\rule\z@\footnotesep\ignorespaces\the\tntoksthree\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\S$}\rule\z@\footnotesep\ignorespaces\the\tntoksfour\@finalstrut\strutbox}% +\fi +\ifnum\titlenotecount=5 + \@maketntext{% + \raisebox{4pt}{$\ast$}\rule\z@\footnotesep\ignorespaces\the\tntoks\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\dagger$}\rule\z@\footnotesep\ignorespaces\the\tntokstwo\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\ddagger$}\rule\z@\footnotesep\ignorespaces\the\tntoksthree\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\S$}\rule\z@\footnotesep\ignorespaces\the\tntoksfour\par\@finalstrut\strutbox}% +\@maketntext{% + \raisebox{4pt}{$\P$}\rule\z@\footnotesep\ignorespaces\the\tntoksfive\@finalstrut\strutbox}% +\fi + \color@endgroup} %g} +\fi +\setcounter{footnote}{0} +\section*{ABSTRACT}\normalsize%\ninept +} + +\def\endabstract{\if@twocolumn\else\endquotation\fi} + +\def\keywords{\if@twocolumn +\section*{Keywords} +\else \small +\quotation +\fi} + +\def\terms{\if@twocolumn +\section*{General Terms} +\else \small +\quotation +\fi} + +% -- Classification needs to be a bit smart due to optionals - Gerry/Georgia November 2nd. 1999 +\newcount\catcount +\global\catcount=1 + +\def\category#1#2#3{% +\ifnum\catcount=1 +\section*{Categories and Subject Descriptors} +\advance\catcount by 1\else{\unskip; }\fi + \@ifnextchar [{\@category{#1}{#2}{#3}}{\@category{#1}{#2}{#3}[]}% +} + +\def\@category#1#2#3[#4]{% + \begingroup + \let\and\relax + #1 [\textbf{#2}]% + \if!#4!% + \if!#3!\else : #3\fi + \else + :\space + \if!#3!\else #3\kern\z@---\hskip\z@\fi + \textit{#4}% + \fi + \endgroup +} +% + +%%% This section (written by KBT) handles the 1" box in the lower left +%%% corner of the left column of the first page by creating a picture, +%%% and inserting the predefined string at the bottom (with a negative +%%% displacement to offset the space allocated for a non-existent +%%% caption). +%%% +\newtoks\copyrightnotice +\def\ftype@copyrightbox{8} +\def\@copyrightspace{ +\@float{copyrightbox}[b] +\begin{center} +\setlength{\unitlength}{1pc} +\begin{picture}(20,6) %Space for copyright notice +\put(0,-0.95){\crnotice{\@toappear}} +\end{picture} +\end{center} +\end@float} + +\def\@toappear{} % Default setting blank - commands below change this. +\long\def\toappear#1{\def\@toappear{\parbox[b]{20pc}{\baselineskip 9pt#1}}} +\def\toappearbox#1{\def\@toappear{\raisebox{5pt}{\framebox[20pc]{\parbox[b]{19pc}{#1}}}}} + +\newtoks\conf +\newtoks\confinfo +\def\conferenceinfo#1#2{\global\conf={#1}\global\confinfo{#2}} + + +%\def\marginpar{\@latexerr{The \marginpar command is not allowed in the +% `acmconf' document style.}\@eha} + +\def\marginpar{\ClassError{% + \string\marginpar\space is not allowed in the `acmconf' document % January 2008 + style}\@eha} + +\mark{{}{}} % Initializes TeX's marks + +\def\today{\ifcase\month\or + January\or February\or March\or April\or May\or June\or + July\or August\or September\or October\or November\or December\fi + \space\number\day, \number\year} + +\def\@begintheorem#1#2{% + \parskip 0pt % GM July 2000 (for tighter spacing) + \trivlist + \item[% + \hskip 10\p@ + \hskip \labelsep + {{\sc #1}\hskip 5\p@\relax#2.}% + ] + \it +} +\def\@opargbegintheorem#1#2#3{% + \parskip 0pt % GM July 2000 (for tighter spacing) + \trivlist + \item[% + \hskip 10\p@ + \hskip \labelsep + {\sc #1\ #2\ % This mod by Gerry to enumerate corollaries + \setbox\@tempboxa\hbox{(#3)} % and bracket the 'corollary title' + \ifdim \wd\@tempboxa>\z@ % and retain the correct numbering of e.g. theorems + \hskip 5\p@\relax % if they occur 'around' said corollaries. + \box\@tempboxa % Gerry - Nov. 1999. + \fi.}% + ] + \it +} +\newif\if@qeded +\global\@qededfalse + +% -- original +%\def\proof{% +% \vspace{-\parskip} % GM July 2000 (for tighter spacing) +% \global\@qededfalse +% \@ifnextchar[{\@xproof}{\@proof}% +%} +% -- end of original + +% (JSS) Fix for vertical spacing bug - Gerry Murray July 30th. 2002 +\def\proof{% +\vspace{-\lastskip}\vspace{-\parsep}\penalty-51% +\global\@qededfalse +\@ifnextchar[{\@xproof}{\@proof}% +} + +\def\endproof{% + \if@qeded\else\qed\fi + \endtrivlist +} +\def\@proof{% + \trivlist + \item[% + \hskip 10\p@ + \hskip \labelsep + {\sc Proof.}% + ] + \ignorespaces +} +\def\@xproof[#1]{% + \trivlist + \item[\hskip 10\p@\hskip \labelsep{\sc Proof #1.}]% + \ignorespaces +} +\def\qed{% + \unskip + \kern 10\p@ + \begingroup + \unitlength\p@ + \linethickness{.4\p@}% + \framebox(6,6){}% + \endgroup + \global\@qededtrue +} + +\def\newdef#1#2{% + \expandafter\@ifdefinable\csname #1\endcsname + {\@definecounter{#1}% + \expandafter\xdef\csname the#1\endcsname{\@thmcounter{#1}}% + \global\@namedef{#1}{\@defthm{#1}{#2}}% + \global\@namedef{end#1}{\@endtheorem}% + }% +} +\def\@defthm#1#2{% + \refstepcounter{#1}% + \@ifnextchar[{\@ydefthm{#1}{#2}}{\@xdefthm{#1}{#2}}% +} +\def\@xdefthm#1#2{% + \@begindef{#2}{\csname the#1\endcsname}% + \ignorespaces +} +\def\@ydefthm#1#2[#3]{% + \trivlist + \item[% + \hskip 10\p@ + \hskip \labelsep + {\it #2% +% \savebox\@tempboxa{#3}% + \saveb@x\@tempboxa{#3}% % January 2008 + \ifdim \wd\@tempboxa>\z@ + \ \box\@tempboxa + \fi.% + }]% + \ignorespaces +} +\def\@begindef#1#2{% + \trivlist + \item[% + \hskip 10\p@ + \hskip \labelsep + {\it #1\ \rm #2.}% + ]% +} +\def\theequation{\arabic{equation}} + +\newcounter{part} +\newcounter{section} +\newcounter{subsection}[section] +\newcounter{subsubsection}[subsection] +\newcounter{paragraph}[subsubsection] +\def\thepart{\Roman{part}} +\def\thesection{\arabic{section}} +\def\thesubsection{\thesection.\arabic{subsection}} +\def\thesubsubsection{\thesubsection.\arabic{subsubsection}} %removed \subsecfnt 29 July 2002 gkmt +\def\theparagraph{\thesubsubsection.\arabic{paragraph}} %removed \subsecfnt 29 July 2002 gkmt +\newif\if@uchead +\@ucheadfalse + +%% CHANGES: NEW NOTE +%% NOTE: OK to use old-style font commands below, since they were +%% suitably redefined for LaTeX2e +%% END CHANGES +\setcounter{secnumdepth}{3} +\def\part{% + \@startsection{part}{9}{\z@}{-10\p@ \@plus -4\p@ \@minus -2\p@} + {4\p@}{\normalsize\@ucheadtrue}% +} +\def\section{% + \@startsection{section}{1}{\z@}{-10\p@ \@plus -4\p@ \@minus -2\p@}% GM + {4\p@}{\baselineskip 14pt\secfnt\@ucheadtrue}% +} + +\def\subsection{% + \@startsection{subsection}{2}{\z@}{-8\p@ \@plus -2\p@ \@minus -\p@} + {4\p@}{\secfnt}% +} +\def\subsubsection{% + \@startsection{subsubsection}{3}{\z@}{-8\p@ \@plus -2\p@ \@minus -\p@}% + {4\p@}{\subsecfnt}% +} +%\def\paragraph{% +% \vskip 12pt\@startsection{paragraph}{3}{\z@}{6\p@ \@plus \p@}% original +% {-5\p@}{\subsecfnt}% +%} +% If one wants sections, subsections and subsubsections numbered, +% but not paragraphs, one usually sets secnumepth to 3. +% For that, the "depth" of paragraphs must be given correctly +% in the definition (``4'' instead of ``3'' as second argument +% of @startsection): +\def\paragraph{% + \vskip 12pt\@startsection{paragraph}{4}{\z@}{6\p@ \@plus \p@}% % GM and Wolfgang May - 11/30/06 + {-5\p@}{\subsecfnt}% +} +\let\@period=. +\def\@startsection#1#2#3#4#5#6{% + \if@noskipsec %gkmt, 11 aug 99 + \global\let\@period\@empty + \leavevmode + \global\let\@period.% + \fi + \par % + \@tempskipa #4\relax + \@afterindenttrue + \ifdim \@tempskipa <\z@ + \@tempskipa -\@tempskipa + \@afterindentfalse + \fi + \if@nobreak + \everypar{}% + \else + \addpenalty\@secpenalty + \addvspace\@tempskipa + \fi +\parskip=0pt % GM July 2000 (non numbered) section heads + \@ifstar + {\@ssect{#3}{#4}{#5}{#6}} + {\@dblarg{\@sect{#1}{#2}{#3}{#4}{#5}{#6}}}% +} +\def\@sect#1#2#3#4#5#6[#7]#8{% + \ifnum #2>\c@secnumdepth + \let\@svsec\@empty + \else + \refstepcounter{#1}% + \edef\@svsec{% + \begingroup + %\ifnum#2>2 \noexpand\rm \fi % changed to next 29 July 2002 gkmt + \ifnum#2>2 \noexpand#6 \fi + \csname the#1\endcsname + \endgroup + \ifnum #2=1\relax .\fi + \hskip 1em + }% + \fi + \@tempskipa #5\relax + \ifdim \@tempskipa>\z@ + \begingroup + #6\relax + \@hangfrom{\hskip #3\relax\@svsec}% + \begingroup + \interlinepenalty \@M + \if@uchead + \uppercase{#8}% + \else + #8% + \fi + \par + \endgroup + \endgroup + \csname #1mark\endcsname{#7}% + \vskip -12pt %gkmt, 11 aug 99 and GM July 2000 (was -14) - numbered section head spacing +\addcontentsline{toc}{#1}{% + \ifnum #2>\c@secnumdepth \else + \protect\numberline{\csname the#1\endcsname}% + \fi + #7% + }% + \else + \def\@svsechd{% + #6% + \hskip #3\relax + \@svsec + \if@uchead + \uppercase{#8}% + \else + #8% + \fi + \csname #1mark\endcsname{#7}% + \addcontentsline{toc}{#1}{% + \ifnum #2>\c@secnumdepth \else + \protect\numberline{\csname the#1\endcsname}% + \fi + #7% + }% + }% + \fi + \@xsect{#5}\hskip 1pt + \par +} +\def\@xsect#1{% + \@tempskipa #1\relax + \ifdim \@tempskipa>\z@ + \par + \nobreak + \vskip \@tempskipa + \@afterheading + \else + \global\@nobreakfalse + \global\@noskipsectrue + \everypar{% + \if@noskipsec + \global\@noskipsecfalse + \clubpenalty\@M + \hskip -\parindent + \begingroup + \@svsechd + \@period + \endgroup + \unskip + \@tempskipa #1\relax + \hskip -\@tempskipa + \else + \clubpenalty \@clubpenalty + \everypar{}% + \fi + }% + \fi + \ignorespaces +} +\def\@trivlist{% + \@topsepadd\topsep + \if@noskipsec + \global\let\@period\@empty + \leavevmode + \global\let\@period.% + \fi + \ifvmode + \advance\@topsepadd\partopsep + \else + \unskip + \par + \fi + \if@inlabel + \@noparitemtrue + \@noparlisttrue + \else + \@noparlistfalse + \@topsep\@topsepadd + \fi + \advance\@topsep \parskip + \leftskip\z@skip + \rightskip\@rightskip + \parfillskip\@flushglue + \@setpar{\if@newlist\else{\@@par}\fi} + \global\@newlisttrue + \@outerparskip\parskip +} + +%%% Actually, 'abbrev' works just fine as the default +%%% Bibliography style. + +\typeout{Using 'Abbrev' bibliography style} +\newcommand\bibyear[2]{% + \unskip\quad\ignorespaces#1\unskip + \if#2..\quad \else \quad#2 \fi +} +\newcommand{\bibemph}[1]{{\em#1}} +\newcommand{\bibemphic}[1]{{\em#1\/}} +\newcommand{\bibsc}[1]{{\sc#1}} +\def\@normalcite{% + \def\@cite##1##2{[##1\if@tempswa , ##2\fi]}% +} +\def\@citeNB{% + \def\@cite##1##2{##1\if@tempswa , ##2\fi}% +} +\def\@citeRB{% + \def\@cite##1##2{##1\if@tempswa , ##2\fi]}% +} +\def\start@cite#1#2{% + \edef\citeauthoryear##1##2##3{% + ###1% + \ifnum#2=\z@ \else\ ###2\fi + }% + \ifnum#1=\thr@@ + \let\@@cite\@citeyear + \else + \let\@@cite\@citenormal + \fi + \@ifstar{\@citeNB\@@cite}{\@normalcite\@@cite}% +} +%\def\cite{\start@cite23} +\DeclareRobustCommand\cite{\start@cite23} % January 2008 +\def\citeNP{\cite*} % No Parentheses e.g. 5 +%\def\citeA{\start@cite10} +\DeclareRobustCommand\citeA{\start@cite10} % January 2008 +\def\citeANP{\citeA*} +%\def\shortcite{\start@cite23} +\DeclareRobustCommand\shortcite{\start@cite23} % January 2008 +\def\shortciteNP{\shortcite*} +%\def\shortciteA{\start@cite20} +\DeclareRobustCommand\shortciteA{\start@cite20} % January 2008 +\def\shortciteANP{\shortciteA*} +%\def\citeyear{\start@cite30} +\DeclareRobustCommand\citeyear{\start@cite30} % January 2008 +\def\citeyearNP{\citeyear*} +%\def\citeN{% +\DeclareRobustCommand\citeN{% % January 2008 + \@citeRB + \def\citeauthoryear##1##2##3{##1\ [##3% + \def\reserved@a{##1}% + \def\citeauthoryear####1####2####3{% + \def\reserved@b{####1}% + \ifx\reserved@a\reserved@b + ####3% + \else + \errmessage{Package acmart Error: author mismatch + in \string\citeN^^J^^J% + See the acmart package documentation for explanation}% + \fi + }% + }% + \@ifstar\@citeyear\@citeyear +} +%\def\shortciteN{% +\DeclareRobustCommand\shortciteN{% % January 2008 + \@citeRB + \def\citeauthoryear##1##2##3{##2\ [##3% + \def\reserved@a{##2}% + \def\citeauthoryear####1####2####3{% + \def\reserved@b{####2}% + \ifx\reserved@a\reserved@b + ####3% + \else + \errmessage{Package acmart Error: author mismatch + in \string\shortciteN^^J^^J% + See the acmart package documentation for explanation}% + \fi + }% + }% + \@ifstar\@citeyear\@citeyear % GM July 2000 +} + +\def\@citenormal{% + \@ifnextchar [{\@tempswatrue\@citex;}% +% original {\@tempswafalse\@citex,[]}% was ; Gerry 2/24/00 +{\@tempswafalse\@citex[]}% % GERRY FIX FOR BABEL 3/20/2009 +} + +\def\@citeyear{% + \@ifnextchar [{\@tempswatrue\@citex,}% +% original {\@tempswafalse\@citex,[]}% +{\@tempswafalse\@citex[]}% % GERRY FIX FOR BABEL 3/20/2009 +} + +\def\@citex#1[#2]#3{% + \let\@citea\@empty + \@cite{% + \@for\@citeb:=#3\do{% + \@citea +% original \def\@citea{#1 }% + \def\@citea{#1, }% % GERRY FIX FOR BABEL 3/20/2009 -- SO THAT YOU GET [1, 2] IN THE BODY TEXT + \edef\@citeb{\expandafter\@iden\@citeb}% + \if@filesw + \immediate\write\@auxout{\string\citation{\@citeb}}% + \fi + \@ifundefined{b@\@citeb}{% + {\bf ?}% + \@warning{% + Citation `\@citeb' on page \thepage\space undefined% + }% + }% + {\csname b@\@citeb\endcsname}% + }% + }{#2}% +} +%\let\@biblabel\@gobble % Dec. 2008 - Gerry +% ---- +\def\@biblabelnum#1{[#1]} % Gerry's solution #1 - for Natbib -- April 2009 +\let\@biblabel=\@biblabelnum % Gerry's solution #1 - for Natbib -- April 2009 +\def\newblock{\relax} % Gerry Dec. 2008 +% --- +\newdimen\bibindent +\setcounter{enumi}{1} +\bibindent=0em +\def\thebibliography#1{% +\ifnum\addauflag=0\addauthorsection\global\addauflag=1\fi + \section[References]{% <=== OPTIONAL ARGUMENT ADDED HERE + {References} % was uppercased but this affects pdf bookmarks (SP/GM October 2004) + {\vskip -9pt plus 1pt} % GM Nov. 2006 / GM July 2000 (for somewhat tighter spacing) + \@mkboth{{\refname}}{{\refname}}% + }% + \list{[\arabic{enumi}]}{% + \settowidth\labelwidth{[#1]}% + \leftmargin\labelwidth + \advance\leftmargin\labelsep + \advance\leftmargin\bibindent + \parsep=0pt\itemsep=1pt % GM July 2000 + \itemindent -\bibindent + \listparindent \itemindent + \usecounter{enumi} + }% + \let\newblock\@empty + \raggedright % GM July 2000 + \sloppy + \sfcode`\.=1000\relax +} + + +\gdef\balancecolumns +{\vfill\eject +\global\@colht=\textheight +\global\ht\@cclv=\textheight +} + +\newcount\colcntr +\global\colcntr=0 +%\newbox\savebox +\newbox\saveb@x % January 2008 + +\gdef \@makecol {% +\global\advance\colcntr by 1 +\ifnum\colcntr>2 \global\colcntr=1\fi + \ifvoid\footins + \setbox\@outputbox \box\@cclv + \else + \setbox\@outputbox \vbox{% +\boxmaxdepth \@maxdepth + \@tempdima\dp\@cclv + \unvbox \@cclv + \vskip-\@tempdima + \vskip \skip\footins + \color@begingroup + \normalcolor + \footnoterule + \unvbox \footins + \color@endgroup + }% + \fi + \xdef\@freelist{\@freelist\@midlist}% + \global \let \@midlist \@empty + \@combinefloats + \ifvbox\@kludgeins + \@makespecialcolbox + \else + \setbox\@outputbox \vbox to\@colht {% +\@texttop + \dimen@ \dp\@outputbox + \unvbox \@outputbox + \vskip -\dimen@ + \@textbottom + }% + \fi + \global \maxdepth \@maxdepth +} +\def\titlenote{\@ifnextchar[\@xtitlenote{\stepcounter\@mpfn +\global\advance\titlenotecount by 1 +\ifnum\titlenotecount=1 + \raisebox{9pt}{$\ast$} +\fi +\ifnum\titlenotecount=2 + \raisebox{9pt}{$\dagger$} +\fi +\ifnum\titlenotecount=3 + \raisebox{9pt}{$\ddagger$} +\fi +\ifnum\titlenotecount=4 +\raisebox{9pt}{$\S$} +\fi +\ifnum\titlenotecount=5 +\raisebox{9pt}{$\P$} +\fi + \@titlenotetext +}} + +\long\def\@titlenotetext#1{\insert\footins{% +\ifnum\titlenotecount=1\global\tntoks={#1}\fi +\ifnum\titlenotecount=2\global\tntokstwo={#1}\fi +\ifnum\titlenotecount=3\global\tntoksthree={#1}\fi +\ifnum\titlenotecount=4\global\tntoksfour={#1}\fi +\ifnum\titlenotecount=5\global\tntoksfive={#1}\fi + \reset@font\footnotesize + \interlinepenalty\interfootnotelinepenalty + \splittopskip\footnotesep + \splitmaxdepth \dp\strutbox \floatingpenalty \@MM + \hsize\columnwidth \@parboxrestore + \protected@edef\@currentlabel{% + }% + \color@begingroup + \color@endgroup}} + +%%%%%%%%%%%%%%%%%%%%%%%%% +\ps@plain +\baselineskip=11pt +\let\thepage\relax % For NO page numbers - GM Nov. 30th. 1999 and July 2000 +\def\setpagenumber#1{\global\setcounter{page}{#1}} +%\pagenumbering{arabic} % Arabic page numbers GM July 2000 +\twocolumn % Double column. +\flushbottom % Even bottom -- alas, does not balance columns at end of document +\pagestyle{plain} + +% Need Copyright Year and Copyright Data to be user definable (in .tex file). +% Gerry Nov. 30th. 1999 +\newtoks\copyrtyr +\newtoks\acmcopyr +\newtoks\boilerplate +\global\acmcopyr={X-XXXXX-XX-X/XX/XX} % Default - 5/11/2001 *** Gerry +\global\copyrtyr={20XX} % Default - 3/3/2003 *** Gerry +\def\CopyrightYear#1{\global\copyrtyr{#1}} +\def\crdata#1{\global\acmcopyr{#1}} +\def\permission#1{\global\boilerplate{#1}} +% +\global\boilerplate={Permission to make digital or hard copies of all or part of this work for personal or classroom use is granted without fee provided that copies are not made or distributed for profit or commercial advantage and that copies bear this notice and the full citation on the first page. To copy otherwise, to republish, to post on servers or to redistribute to lists, requires prior specific permission and/or a fee.} +\newtoks\copyrightetc +\global\copyrightetc{Copyright \the\copyrtyr\ ACM \the\acmcopyr\ ...\$15.00} % Gerry changed to 15 May 2012 +\toappear{\the\boilerplate\par +{\confname{\the\conf}} \the\confinfo\par \the\copyrightetc.} +%\DeclareFixedFont{\altcrnotice}{OT1}{tmr}{m}{n}{8} % << patch needed for accenting e.g. Montreal - Gerry, May 2007 +%\DeclareFixedFont{\altconfname}{OT1}{tmr}{m}{it}{8} % << patch needed for accenting in italicized confname - Gerry, May 2007 +% +%{\altconfname{{\the\conf}}} {\altcrnotice\the\confinfo\par} \the\copyrightetc.} % << Gerry, May 2007 +% +% The following section (i.e. 3 .sty inclusions) was added in May 2007 so as to fix the problems that many +% authors were having with accents. Sometimes accents would occur, but the letter-character would be of a different +% font. Conversely the letter-character font would be correct but, e.g. a 'bar' would appear superimposed on the +% character instead of, say, an unlaut/diaresis. Sometimes the letter-character would NOT appear at all. +% Using [T1]{fontenc} outright was not an option as this caused 99% of the authors to 'produce' a Type-3 (bitmapped) +% PDF file - useless for production. +% +% For proper (font) accenting we NEED these packages to be part of the .cls file i.e. 'ae', 'aecompl' and 'aeguil' +% ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +%% This is file `ae.sty' +\def\fileversion{1.3} +\def\filedate{2001/02/12} +\NeedsTeXFormat{LaTeX2e} +%\ProvidesPackage{ae}[\filedate\space\fileversion\space % GM +% Almost European Computer Modern] % GM - keeping the log file clean(er) +\newif\if@ae@slides \@ae@slidesfalse +\DeclareOption{slides}{\@ae@slidestrue} +\ProcessOptions +\fontfamily{aer} +\RequirePackage[T1]{fontenc} +\if@ae@slides + \renewcommand{\sfdefault}{laess} + \renewcommand{\rmdefault}{laess} % no roman + \renewcommand{\ttdefault}{laett} +\else + \renewcommand{\sfdefault}{aess} + \renewcommand{\rmdefault}{aer} + \renewcommand{\ttdefault}{aett} +\fi +\endinput +%% +%% End of file `ae.sty'. +% +% +\def\fileversion{0.9} +\def\filedate{1998/07/23} +\NeedsTeXFormat{LaTeX2e} +%\ProvidesPackage{aecompl}[\filedate\space\fileversion\space % GM +%T1 Complements for AE fonts (D. Roegel)] % GM -- keeping the log file clean(er) + +\def\@ae@compl#1{{\fontencoding{T1}\fontfamily{cmr}\selectfont\symbol{#1}}} +\def\guillemotleft{\@ae@compl{19}} +\def\guillemotright{\@ae@compl{20}} +\def\guilsinglleft{\@ae@compl{14}} +\def\guilsinglright{\@ae@compl{15}} +\def\TH{\@ae@compl{222}} +\def\NG{\@ae@compl{141}} +\def\ng{\@ae@compl{173}} +\def\th{\@ae@compl{254}} +\def\DJ{\@ae@compl{208}} +\def\dj{\@ae@compl{158}} +\def\DH{\@ae@compl{208}} +\def\dh{\@ae@compl{240}} +\def\@perthousandzero{\@ae@compl{24}} +\def\textperthousand{\%\@perthousandzero} +\def\textpertenthousand{\%\@perthousandzero\@perthousandzero} +\endinput +% +% +%% This is file `aeguill.sty' +% This file gives french guillemets (and not guillemots!) +% built with the Polish CMR fonts (default), WNCYR fonts, the LASY fonts +% or with the EC fonts. +% This is useful in conjunction with the ae package +% (this package loads the ae package in case it has not been loaded) +% and with or without the french(le) package. +% +% In order to get the guillemets, it is necessary to either type +% \guillemotleft and \guillemotright, or to use an 8 bit encoding +% (such as ISO-Latin1) which selects these two commands, +% or, if you use the french package (but not the frenchle package), +% to type << or >>. +% +% By default, you get the Polish CMR guillemets; if this package is loaded +% with the `cm' option, you get the LASY guillemets; with `ec,' you +% get the EC guillemets, and with `cyr,' you get the cyrillic guillemets. +% +% In verbatim mode, you always get the EC/TT guillemets. +% +% The default option is interesting in conjunction with PDF, +% because there is a Type 1 version of the Polish CMR fonts +% and these guillemets are very close in shape to the EC guillemets. +% There are no free Type 1 versions of the EC fonts. +% +% Support for Polish CMR guillemets was kindly provided by +% Rolf Niepraschk in version 0.99 (2000/05/22). +% Bernd Raichle provided extensive simplifications to the code +% for version 1.00. +% +% This package is released under the LPPL. +% +% Changes: +% Date version +% 2001/04/12 1.01 the frenchle and french package are now distinguished. +% +\def\fileversion{1.01} +\def\filedate{2001/04/12} +\NeedsTeXFormat{LaTeX2e} +%\ProvidesPackage{aeguill}[2001/04/12 1.01 % % GM +%AE fonts with french guillemets (D. Roegel)] % GM - keeping the log file clean(er) +%\RequirePackage{ae} % GM May 2007 - already embedded here + +\newcommand{\@ae@switch}[4]{#4} +\DeclareOption{ec}{\renewcommand\@ae@switch[4]{#1}} +\DeclareOption{cm}{\renewcommand\@ae@switch[4]{#2}} +\DeclareOption{cyr}{\renewcommand\@ae@switch[4]{#3}} +\DeclareOption{pl}{\renewcommand\@ae@switch[4]{#4}} +\ExecuteOptions{pl} +\ProcessOptions + +% +% Load necessary packages +% +\@ae@switch{% ec + % do nothing +}{% cm + \RequirePackage{latexsym}% GM - May 2007 - already 'mentioned as required' up above +}{% cyr + \RequirePackage[OT2,T1]{fontenc}% +}{% pl + \RequirePackage[OT4,T1]{fontenc}% +} + +% The following command will be compared to \frenchname, +% as defined in french.sty and frenchle.sty. +\def\aeguillfrenchdefault{french}% + +\let\guill@verbatim@font\verbatim@font +\def\verbatim@font{\guill@verbatim@font\ecguills{cmtt}% + \let\guillemotleft\@oguills\let\guillemotright\@fguills} + +\begingroup \catcode`\<=13 \catcode`\>=13 +\def\x{\endgroup + \def\ae@lfguill{<<}% + \def\ae@rfguill{>>}% +}\x + +\newcommand{\ecguills}[1]{% + \def\selectguillfont{\fontencoding{T1}\fontfamily{#1}\selectfont}% + \def\@oguills{{\selectguillfont\symbol{19}}}% + \def\@fguills{{\selectguillfont\symbol{20}}}% + } + +\newcommand{\aeguills}{% + \ae@guills + % We redefine \guillemotleft and \guillemotright + % in order to catch them when they are used + % with \DeclareInputText (in latin1.def for instance) + % We use \auxWARNINGi as a safe indicator that french.sty is used. + \gdef\guillemotleft{\ifx\auxWARNINGi\undefined + \@oguills % neither french.sty nor frenchle.sty + \else + \ifx\aeguillfrenchdefault\frenchname + \ae@lfguill % french.sty + \else + \@oguills % frenchle.sty + \fi + \fi}% + \gdef\guillemotright{\ifx\auxWARNINGi\undefined + \@fguills % neither french.sty nor frenchle.sty + \else + \ifx\aeguillfrenchdefault\frenchname + \ae@rfguill % french.sty + \else + \@fguills % frenchle.sty + \fi + \fi}% + } + +% +% Depending on the class option +% define the internal command \ae@guills +\@ae@switch{% ec + \newcommand{\ae@guills}{% + \ecguills{cmr}}% +}{% cm + \newcommand{\ae@guills}{% + \def\selectguillfont{\fontencoding{U}\fontfamily{lasy}% + \fontseries{m}\fontshape{n}\selectfont}% + \def\@oguills{\leavevmode\nobreak + \hbox{\selectguillfont (\kern-.20em(\kern.20em}\nobreak}% + \def\@fguills{\leavevmode\nobreak + \hbox{\selectguillfont \kern.20em)\kern-.2em)}% + \ifdim\fontdimen\@ne\font>\z@\/\fi}}% +}{% cyr + \newcommand{\ae@guills}{% + \def\selectguillfont{\fontencoding{OT2}\fontfamily{wncyr}\selectfont}% + \def\@oguills{{\selectguillfont\symbol{60}}}% + \def\@fguills{{\selectguillfont\symbol{62}}}} +}{% pl + \newcommand{\ae@guills}{% + \def\selectguillfont{\fontencoding{OT4}\fontfamily{cmr}\selectfont}% + \def\@oguills{{\selectguillfont\symbol{174}}}% + \def\@fguills{{\selectguillfont\symbol{175}}}} +} + + +\AtBeginDocument{% + \ifx\GOfrench\undefined + \aeguills + \else + \let\aeguill@GOfrench\GOfrench + \gdef\GOfrench{\aeguill@GOfrench \aeguills}% + \fi + } + +\endinput +% + diff --git a/correctness-model/writeup/speccfg.tex b/correctness-model/writeup/speccfg.tex new file mode 100644 index 0000000..661c778 --- /dev/null +++ b/correctness-model/writeup/speccfg.tex @@ -0,0 +1,17 @@ +\begin{figure} + + +%\paragraph{Default settings} +%\begin{grammar} + +% ::= `=' +%\alt `for' `=' `to' `do' +%\alt `{' `}' +%\alt + +% ::= `;' | + +%\end{grammar} + +\caption{\label{fig:algorithm}\TOOL algorithm} +\end{figure} diff --git a/correctness-model/writeup/specification.tex b/correctness-model/writeup/specification.tex new file mode 100644 index 0000000..6fe0447 --- /dev/null +++ b/correctness-model/writeup/specification.tex @@ -0,0 +1,416 @@ +\mysection{Specification Language Design}\label{sec:specification} + +We begin by overviewing \TOOL's basic approach. The \TOOL +specification language specifies the correctness properties for +concurrent data structures by establishing a correspondence with an +equivalent sequential data structure, which requires: (1) defining the +equivalent sequential data structure; (2) establishing an equivalent +sequential history; (3) relating the behavior of the concurrent +execution to the sequential history; and (4) specifying the +synchronization properties between method invocations. The +specification language has the following key components: + +\mypara{\bf 1. Equivalent Sequential Data Structure:} This component defines a +sequential data structure against which the concurrent data structure +will be checked. + +\mypara{\bf 2. Defining the Equivalent Sequential History:} +For real-world data structures, generally there exist two types of API methods, +\textit{primitive} API methods and \textit{aggregate} API methods. Take +concurrent hashtables as an example, it provides primitive API methods +such as \code{put} and \code{get} that implement the core functionality of the +data structure, and it also has aggregate API methods such +as \code{putAll} that calls primitive API methods internally. From our +experience working with our benchmark set, it is generally possible to +generate a sequential history of primitive API method invocations for +real-world data structures as they generally serialize on a single memory +location --- while it is possible to observe partially completed aggregate +API method invocations. Therefore, our specifications will focus on the +correctness of primitive API methods. + +Borrowing from VYRD~\cite{vyrd} the concept of commit points, we allow +developers to specify \textit{ordering points} --- the specific atomic +operations between method invocation and response events that are used for +ordering method calls. Ordering points are a generalization of the commit +points as they also use memory operations that do not commit the data structure +operation to order method calls. Developers then specify ordering points to +generate the equivalent sequential history. + +\mypara{\bf 3. Specifying Correct Behaviors:} Developers use +the \TOOL specification to provide a set of \textit{side effects} +and \textit{assertions} to describe the desired behavior of each API +method. Side effects capture how a method updates the equivalent +sequential data structure. Assertions include both preconditions and +postconditions which specify what conditions each method must satisfy +before and after the method call, respectively. The specification of +the side effects and assertions may make use of the states of the +equivalent sequential data structure, meaning that these components +can access the internal variables and methods of the equivalent sequential data +structure. Additionally, they can reference the values available at +the method invocation and response, i.e., the parameter values and the +return value. + +\mypara{\bf 4. Synchronization:} The synchronization specification +describes the desired happens before relations between API method +invocations. \TOOL specifications allow developers to specify the +properties at the abstraction of methods. This makes specifications +cleaner, more understandable, and less error-prone because the +properties do not rely on low-level implementation +details. Moreover, \TOOL specifications allow developers to attach +conditions to methods when specifying synchronization properties so +that a method call might synchronize with another only under a +specific condition. For example, consider a spin lock with +a \code{try\_lock()} method, we need to specify that only a +successful \code{try\_lock()} method invocation must synchronize with the +previous \code{unlock()} method invocation. + +Figure~\ref{fig:speccfg} presents the grammar for the \TOOL specification +language. The grammar defines three types of specification annotations: +\textit{structure annotations}, \textit{method annotations}, and +\textit{ordering point annotations}. +In the grammar, \textit{code} means legal C/C++ source code; and +\textit{label} means a legal C/C++ variable name. +Annotations are embedded in C/C++ comments. +This format does not affect the semantics of the original +program and allows for the same source to be used by both a standard C/C++ +compiler to generate production code and for the \TOOL specification compiler to +extract the \TOOL specification from the comments. We discuss these constructs in more detail throughout the remainder of this section. + +\begin{figure}[!tb] +\vspace{-.2cm} +{\scriptsize + \begin{eqnarray*} + \textit{\textbf{structureSpec}} & \rightarrow & \code{"@Structure\_Define"}\\ + && \textit{structureDefine} \ \ \ (\textit{happensBefore})?\\ + \textit{structureDefine} & \rightarrow & +(\code{"@DeclareStruct:"} \ + \textit{code})\text{*}\\ + && \code{"@DeclareVar:"} \ \textit{code}\\ + && \code{"@InitVar:"} \ \textit{code}\\ + && (\code{"@DefineFunc:"} \ \textit{code})\text{*}\\ + \textit{happensBefore} & \rightarrow & \code{"@Happens\_Before:"} \ + (\textit{invocation} \ \code{"->"} \ + \textit{invocation})\textsuperscript{+}\\ + \textit{invocation} & \rightarrow & \textit{label} \ (\ \ \code{"("} \ + \textit{label} \code{")"} \ \ )?\\ + \textit{\textbf{methodSpec}} & \rightarrow & \code{"@Method:"} \ \textit{label}\\ + && \code{"@Ordering\_Point\_Set:"} \ \textit{label} \ (\code{"|"} \ + \textit{label})\text{*}\\ + && (\code{"@HB\_Condition:"} \ \textit{label} \ \code{"::"} \ \textit{code})\text{*}\\ + && (\code{"@ID:"} \ \textit{code})?\\ + && (\code{"@PreCondition:"} \ \textit{code})?\\ + && (\code{"@SideEffect:"} \ \textit{code})?\\ + && (\code{"@PostCondition:"} \ \textit{code})?\\ + \textit{\textbf{potentialOP}} & \rightarrow & \code{"@Potential\_Ordering\_Point:"} \ \textit{code}\\ + && \code{"@Label:"} \ \ \textit{label}\\ + \textit{\textbf{opCheck}} & \rightarrow & \code{"@Ordering\_Point\_Check:"} \ \textit{code}\\ + && \code{"@Potential\_Ordering\_Point\_Label:"} \ \textit{label}\\ + && \code{"@Label:"} \ \ \textit{label}\\ + \textit{\textbf{opLabelCheck}} & \rightarrow & \code{"@Ordering\_Point\_Label\_Check:"} \ \textit{code}\\ + && \code{"@Label:"} \ \ \textit{label}\\ + \textit{\textbf{opClear}} & \rightarrow & \code{"@Ordering\_Point\_Clear:"} \ \textit{code}\\ + && \code{"@Label:"} \ \ \textit{label}\\ + \textit{code} & \rightarrow & \code{} + \end{eqnarray*}} +\vspace{-.7cm} +\caption{\label{fig:speccfg}Grammar for \TOOL specification language} +\vspace{-.3cm} +\end{figure} + + +\mysubsection{Example} + +To make the \TOOL specification language more concrete, +Figure~\ref{fig:rcuSpecExample} presents the annotated RCU example. We +use ordering points to order the method invocations to construct +the equivalent sequential history for the concurrent execution. We +first specify the ordering points for the \code{read} +and \code{update} methods to define the equivalent sequential +history. For the +\code{read} method, the load access of the \code{node} field in +Line~\ref{line:rcuSpecReadLoad} is the only operation that can be an ordering +point. For the \code{update} method, there exists more than one atomic +operation. However, they only serve as an ordering point when it successfully uses the CAS operation +to update the \code{node} field. Thus, we specify in +Line~\ref{line:rcuSpecUpdateOPBegin} that the CAS operation should be the +ordering point for the update operation when \code{succ} is \code{true}. We +associate the methods with the two ordering points in +Lines~\ref{line:rcuSpecReadInterfaceOPSet} and +\ref{line:rcuSpecUpdateInterfaceOPSet}. + +Next, we specify the equivalent sequential data structure for this RCU implementation. +Line~\ref{line:declVar} declares two integer fields \code{\_data} and +\code{\_version} as the internal states of the equivalent sequential RCU, and +Line~\ref{line:initVar} initializes both variables to \code{0}. + +We then specify the correct behaviors of the equivalent sequential +history by defining the side effects and assertions for +the \code{read} and \code{update} methods. Side effects specify how +to perform the corresponding operation on the equivalent sequential +data structure. For example, +Line~\ref{line:rcuSpecUpdateInterfaceSideEffect} specifies the side +effects of the \code{update} to the sequential states. +Assertions specify properties that should be true of the concurrent data structure execution. +For example, Line~\ref{line:rcuSpecReadInterfacePostCondition} +specifies that the \code{read} method should satisfy the postcondition that the +returned \code{data} and \code{version} fields should have consistent values +as the internal states of the equivalent sequential data structure. + +Our implementation of the RCU data structure is designed to establish +synchronization between the \code{update} method invocation and the later +\code{read} or \code{update} method invocation. This is important, for +example, if a client thread were to update an array index and use the RCU data +structure to communicate the updated index to a second client thread. Without +synchronization, the second client thread may not see the update to the array +index. Besides, the synchronization between two \code{update} calls ensures that +a later \code{read} will see the most updated values. +In order to specify the synchronization, we associate \code{read} and \code{update} methods with method call identifiers +(Lines~\ref{line:rcuSpecReadInterfaceID} and~\ref{line:rcuSpecUpdateInterfaceID}), which for this example is the value of the +\code{this} pointer. +Together these annotations ensure that every +API method call on the same RCU object will have the same method call ID. +Line~\ref{line:rcuSpecHB} then specifies +the synchronization properties for the RCU --- any \code{update} method invocation +should synchronize with all later \code{read} and \code{update} method invocations that have the +same method call identifier as that \code{update} method. This guarantees that +\code{update} calls should synchronize with the \code{read} and \code{update} +calls of the same RCU object. + + +\begin{figure}[h!] +\vspace{-.2cm} +\begin{lstlisting}[xleftmargin=6.0ex] +class RCU { + /** @Structure_Define: + @DeclareVar:/*@ \label{line:declVar} @*/ int _data, _version; + @InitVar:/*@ \label{line:initVar} @*/ _data = 0; _version = 0; + @Happens_Before: Update->Read Update->Update*//*@ \label{line:rcuSpecHB} @*/ + atomic node; + public: + RCU() { + Node *n = new Node; + n->data = 0; + n->version = 0; + atomic_init(&node, n); + } + /** @Interface: Read/*@ \label{line:rcuSpecReadInterfaceLabel} @*/ + @Ordering_Point_Set: Read_Point/*@ \label{line:rcuSpecReadInterfaceOPSet} @*/ + @ID: this/*@ \label{line:rcuSpecReadInterfaceID} @*/ + @PostCondition:/*@ \label{line:rcuSpecReadInterfacePostCondition} @*/ + _data == *data && _version == *version *//*@ \label{line:rcuSpecReadInterfaceEnd} @*/ + void read(int *data, int *version) { + Node *res = node.load(mo_acquire);/*@ \label{line:rcuSpecReadLoad} @*/ + /** @Ordering_Point_Label_Check: true/*@ \label{line:rcuSpecReadOPBegin} @*/ + @Label: Read_Point */ + *data = res->data; + *version = res->version; + } + /** @Interface: Update + @Ordering_Point_Set: Update_Point/*@ \label{line:rcuSpecUpdateInterfaceOPSet} @*/ + @ID: this/*@ \label{line:rcuSpecUpdateInterfaceID} @*/ + @SideEffect: _data = data; _version++; *//*@ \label{line:rcuSpecUpdateInterfaceSideEffect} @*/ + void update(int data) { + bool succ = false; + Node *newNode = new Node;/*@ \label{line:rcuSpecUpdateAlloc} @*/ + Node *prev = node.load(mo_acquire);/*@ \label{line:rcuSpecUpdateLoad} @*/ + do { + newNode->data = data; + newNode->version = prev->version + 1; + succ = node.compare_exchange_strong(prev,/*@\label{line:rcuSpecUpdateCAS} @*/ + newNode, mo_acq_rel, mo_acquire); + /** @Ordering_Point_Label_Check: succ/*@ \label{line:rcuSpecUpdateOPBegin} @*/ + @Label: Update_Point */ + } while (!succ); + } +}; + +\end{lstlisting} +\vspace{-.2cm} +\caption{\label{fig:rcuSpecExample}Annotated RCU specification example} +\vspace{-.3cm} +\end{figure} + +\subsection{Defining the Equivalent Sequential History} + +While defining the equivalent sequential history for concurrent executions is well studied in the context of the +SC memory model, optimizations that developers typically use in the +context of weaker memory models create the following new challenges when +we generate the equivalent sequential history using ordering points. + +\squishlist +\item {\bf Absence of a Meaningful Trace or Total Order:} +For the SC memory model, an execution can be represented by a simple +interleaving of all the memory operations, where each load operation reads from +the last store operation to the same location in the trace. However, under a +relaxed memory model like C/C++11, the interleaving does not uniquely define an +execution because a given load operation can potentially read from many +different store operations in the interleaving. Therefore, we have to rely on +the intrinsic relations between ordering points such as \textit{reads from} and +\textit{modification order} to order method calls. + +\item {\bf Lack of Program Order Preserving Sequential History:} +Moreover, as discussed in Section~\ref{sec:exampleMotivation}, in +general it is not possible to arrange an execution in any totally +ordered sequential history that preserves program order. + +A key insight is that many concurrent data structures' API methods +have a commit point, which is a single memory operation that makes the +update visible to other threads and that also serves as an ordering +point. When two data structure operations have a dependence, it is +often the case that their respective commit points are both +conflicting accesses to the same memory location. In this case, the +modification order provided by C/C++ is sufficient to order these +operations since modification order is guaranteed to be consistent +with the happens before relation (and therefore also the sequenced +before relation). + +For cases where the method calls are independent, such as +a \code{put(X, 1)} followed by a +\code{get(Y)} in a hashtable where \code{X} and \code{Y} are different keys, +the lack of an ordering is not a problem since those methods commute. +\squishend + +\mypara{\bf Ordering Points Annotations:} In many cases, it is not +possible to determine whether a given atomic operation is an ordering +point until later in the execution. For example, +some internal methods may be called by multiple API methods. In this +case, the same atomic operation can be identified as a potential +ordering point for multiple API methods, and each API method later has +a checking annotation to verify whether it was a real ordering +point. Therefore, the \TOOL specification separates the definition of ordering +points as follows: +\squishcount +\item {\code{Potential\_Ordering\_Point} annotation:} The labeling of ordering +points that identifies the location of a potential ordering point. +\item {\code{Ordering\_Point\_Check} annotation:} The checking of ordering +points that checks at a later point whether a potential ordering point was +really an ordering point. +\countend + +These two constructs together identify ordering points. For +example, assume that $A$ is an atomic operation that is potentially an +ordering point under some specific +condition. The developer would then write a +\code{Potential\_Ordering\_Point} annotation with a condition +\code{ConditionA} and a label +\code{LabelA}, and then use the label \code{LabelA} in an +\code{Ordering\_Point\_Check} annotation at a later point. + +The \code{Ordering\_Point\_Label\_Check} annotation combines +the \code{Potential\_Ordering\_Point} and the +\code{Ordering\_Point\_Check} annotations, and makes +specifications simpler for the use case where the ordering point is known immediately. +For example, in Line~\ref{line:rcuSpecReadOPBegin} of +Figure~\ref{fig:rcuSpecExample}, we use the \code{Ordering\_Point\_Label\_Check} annotation to +identify the ordering point for \code{read} because we know the load operation +in Line~\ref{line:rcuSpecReadLoad} is an ordering point at the time it is +executed. + +Some data structure operations may require multiple ordering points. +For example, consider a transaction implementation that first attempts +to lock all of the involved objects (dropping the locks and retrying +if it fails to acquire a lock), performs the updates, and then releases +the locks. To order such transactions in a relaxed memory model, we must consider all of the +locks it acquires rather than just the last lock. Thus, we allow a +method invocation to have more than one ordering point, and the +additional ordering points serve to order the operation with respect to +multiple different memory locations. For the transaction example, it +may be necessary to retry the acquisition of locks. To support this +scenario, the \code{Ordering\_Point\_Clear} annotation removes all +previous ordering points when it satisfies a specific condition. + +Moreover, when an API method calls another API method, they can share ordering points. In +that case, \TOOL requires that at that ordering point, the concurrent data +structure should satisfy the precondition and postcondition of both API methods. + +\subsection{Checking the IO Behavior of Methods} + +With the specified ordering points, \TOOL is able to generate the equivalent +sequential history. Developers then need to define the equivalent sequential +data structure. For example, in Line~\ref{line:declVar} and~\ref{line:initVar} of the annotated RCU example, we use the +\code{Structure\_define} annotation to define the equivalent sequential RCU +by specifying the internal states as two integers, \code{\_version} and \code{\_data}. In the +\code{Structure\_define} annotation, developers can also specify definitions for customized +structs and methods for convenience. We design these annotations in such a way +that developers can +write specifications in C/C++ code such that they do not have to learn a new +specification language. + +After defining the internal states and methods of the +equivalent data structure, developers use the \code{SideEffect} annotation to +define the corresponding sequential API methods, which should contain the action +to be performed on the equivalent sequential data structure. For example, in +Line~\ref{line:rcuSpecUpdateInterfaceSideEffect} of the annotated RCU example, +we use \code{SideEffect} to specify that when we execute the \code{update} +method on the equivalent sequential RCU, we should update the +internal states of \code{\_version} and \code{\_data} accordingly. When +the \code{SideEffect} annotation is omitted for a specific API method, it means +that no side effects will happen on the sequential data structure when that +method is called. Take the annotated RCU as an example, the \code{read} has no +side effects on the equivalent sequential RCU. + +With the well-defined equivalent sequential data structure, developers then relate +the generated equivalent sequential history to the equivalent sequential data +structure. In \TOOL, we allow developers to accomplish this by using the +\code{PreCondition} and \code{PostCondition} annotations to specify the +conditions to be checked before and after the API method appears to happen. +For example, Line~\ref{line:rcuSpecReadInterfaceEnd} in the +annotated RCU example means that when \code{read} appears to happen, it should +return the same value as the current internal variables of the equivalent +sequential RCU. +Note that these two annotations contain legitimate C/C++ +expressions that only access the method call parameters, return value and the +internal states of the equivalent sequential data structure. + +\subsection{Checking Synchronization} + +Under a relaxed memory model, compilers and processors can reorder +memory operations and thus the execution can exhibit counter-intuitive +behaviors. The C/C++11 memory model provides developers with memory ordering that establish synchronization, e.g., \code{acquire}, \code{release}, \code{seq\_cst}. Synchronization +serves to control which reorderings are allowed --- however, +restricting reorderings comes at a runtime cost so developers must +balance complexity against runtime overhead. Checking that data +structures establish proper synchronization is important to ensure +that the data structures can be effectively composed with client code. + +We generalize the notion of happens before to methods as follows. +Method call $c_1$ happens-before method call $c_2$ if the invocation +event of $c_1$ happens before the response event of $c_2$. Note that +by this definition two method calls can both happen before each other +--- an example of this is the \code{barrier} synchronization +construct. With this notion, for example, for a correctly synchronized queue, we want +an enqueue to happen before the corresponding dequeue, which avoids +the synchronization problems discussed earlier in +Section~\ref{sec:introNewChallenges}. + +In order to flexibly express the synchronization between methods, we associate +API methods with method call identifiers (or IDs) and happens-before conditional guard +expressions. The method +call ID is a C/C++ expression that computes a unique ID for the call, and if it +is omitted, a default value is used. For example, in our RCU example in +Figure~\ref{fig:rcuSpecExample}, both the \code{update} and \code{read} methods +have the \code{this} pointer of the corresponding RCU object. +The \code{HB\_Condition} component +associates one happens-before conditional guard expression with a unique label. +For one method, multiple conditional guard expressions are allowed to be +defined, and the conditional guard expression can only access the method +instance's argument values and return value. + +After specifying the +method call IDs and the \code{HB\_Condition} labels, developers +can specify the synchronization as ``\code{\small{method1(HB\_condition1)}} +\code{\small{->}} +\code{\small{method2(HB\_condition2)}}''. When the \code{HB\_condition} is omitted, it +defaults to \code{true}. The semantics of this expression is that all +instances of calls to \code{method1} that satisfy the conditional guard +expression \code{HB\_condition1} should happen-before all later instances (as determined by ordering points) of +calls to \code{method2} that satisfy the conditional guard expression +\code{HB\_condition2} such that both instances shared the same ID. +The ID and happens-before conditional guard expression are important because they +allow developers to impose synchronization only between specific method +invocations under specific conditions. For example, in +Figure~\ref{fig:rcuSpecExample}, Line~\ref{line:rcuSpecHB} specifies two +synchronization rules, which together mean that the +\code{update} should only establish synchronization with later \code{read} and +\code{update} from +the same RCU object under all circumstances. diff --git a/correctness-model/writeup/spell.lst b/correctness-model/writeup/spell.lst new file mode 100644 index 0000000..cc7a0be --- /dev/null +++ b/correctness-model/writeup/spell.lst @@ -0,0 +1,99 @@ +personal_ws-1.1 en 98 +runtime +API +dequeuing +opCheck +overviewing +atomics +PSO +thrd +IRIW +optimizations +atomicity +potentialOP +interleavings +updaters +Paraglider +hb +structureSpec +lookups +Lev +enqueue +resizing +linearizability +lrrr +Amit +cnt +MCS +InitVar +bool +multi +spinlock +trylock +Mellor +decrement +Valeadis +cst +happensBefore +SideEffect +putAll +dequeues +DeclareStruct +MPMC +deque +DefineFunc +methodSpec +STL +lockfree +opLabelCheck +rf +structureDefine +SPSC +sb +sc +hashtables +Cppmem +prev +dequeue +reorderings +VYRD +RW +init +TSO +ConditionA +Crummey +sw +opClear +ccccc +Valeiadis +Relacy +Concurrit +postconditions +iteratively +NDetermin +LabelA +resize +succ +TVLA +acq +nondeterministic +intra +linearizable +PreCondition +DeclareVar +hashtable +boolean +postcondition +PostCondition +CheckFence +enqueues +structs +scalability +struct +opo +newNode +linearization +CAS +RCU +Colvin +dependences diff --git a/correctness-model/writeup/technical.tex b/correctness-model/writeup/technical.tex new file mode 100644 index 0000000..674e37c --- /dev/null +++ b/correctness-model/writeup/technical.tex @@ -0,0 +1 @@ +\section{Technical}\label{sec:technical}