Abstract-We consider lightweight usage of model-checking for the debugging of Simulink models. A problem is that modelcheckers typically return only one counterexample, which may slow down the debugging process. We propose an approach and a tool to produce several counterexamples, exemplifying different property violation patterns for a given version of the design. The approach uses data collected during the replay of the counterexamples to synthesize queries for the model-checker, so that it finds counterexamples that activate new paths. The approach is applied to an academic example and an industrial model from the automotive domain.
I. INTRODUCTION
The development of reactive systems commonly uses a model-based approach. The focus is then on designing models from which a prototype or even the real implementation code can be automatically derived. Among the available modeling formalisms, synchronous data-flow languages like Scade [1] or Simulink [2] are well-suited for capturing a reactive behavior. Such a behavior is abstracted in terms of periodically repeated execution cycles in which the system reads inputs from the environment and reacts by emitting outputs. Model simulation facilities allow a thorough testing of the design.
In addition to simulation, Scade and Simulink toolsets also support model-checking, opening the possibility for a formal analysis of models. This paper considers a lightweight approach, in which designers use model-checking as a debugging facility. Analysis is focused on a few critical functions, typically functions synthesizing alarms or elaborating critical Boolean control parameters. The counterexamples returned by the model-checker are used to revise the model under design until it matches the specified property. Our previous work in the avionics domain suggests that this lightweight approach may indeed be quite effective, even with respect to subtle design flaws that are found by lab tests (i.e., tests of the real equipment in the loop with a flight simulator) [3] .
The implementation of the debugging approach may however face several practical problems. One is how to proceed when a lengthy counterexample is returned and the reasons for property violation must be extracted from it. We developed a tool, STANCE (Structural Analysis of counterexamples), to alleviate the effort of engineers [4] . STANCE is interfaced with the Simulink toolset. It automatically extracts the subset of time-stamped input values that are sufficient to reproduce the violation, and visually shows the structural parts of the models that are activated by them and play a role in the observed violation. The tool thus filters out irrelevant data from the counterexample, while allowing engineers to focus on the interesting details of its execution.
A second practical problem is that model-checkers typically return only one counterexample, and no further feedback can be obtained until the model (or the property) is modified to eliminate this counterexample. But there can be other unrevealed flaws in the current version of the model, or the revealed flaw may induce alternative ways to violate the property, which are not shown by the first counterexample found. To speed up the debugging process, it would be more efficient to give the engineer as much feedback as possible before she reworks the design. It calls for forcing the model-checker to search for several counterexamples, that exemplify different property violation patterns for a given version of the design. This is the issue addressed in this paper.
Our search for new counterexamples is based on the structural analysis approach implemented in STANCE. It is inspired by the principle of concolic testing [5] [6] . Concolic testing uses concrete data, collected during the execution of test cases, to guide the constraint-based generation of test cases activating new paths. In a similar way, our approach uses data collected during the replay of the counterexamples to synthesize queries for the model-checker, so that it finds counterexamples that activate new paths. This paper explains how the candidate new paths are extracted from the replay data, and how some automatically inserted instrumentations force the model-checker to consider these paths.
Section II provides background about STANCE and the structural analysis of counterexamples. It also introduces a small case study that we use as a running example to illustrate the concepts throughout the paper. Section III presents the search for new counterexamples, and Section IV illustrates its application to the running example. Section V reports on a case study of larger size than the running example, coming from the automotive industry. Section VI presents related work and Section VII concludes the paper. Figure 1 shows an exemplary verification architecture in Simulink. The model under verification (MUV) is monitored by an observer of the property to check. Both the property and the model are expressed in the same language, i.e. the Simulink data-flow language. This language offers basic operators, like Boolean operators (AND, NOT, ...) and the unit delay 1/z. Besides these basic operators, it is usually convenient to define a library of reusable compound operators, the semantics of which can be given in terms of the basic ones. The expression of the property in Figure 1 uses one compound logical operator (Implies). It also uses two compound temporal operators: a pulse, detecting a falling edge when its Boolean input goes from 1 to 0, and an RS-latch used to store Boolean information. The latch is controlled via its Set and Reset inputs (the star annotation indicates that Reset has priority over Set). Source and sink operators are used for the production of inputs (e.g., the production of action and active) and for the consumption of the final output (e.g., the final value P of the property observer after the logical implication). This output should always have the value 1. The model-checker is then challenged to falsify it. If it succeeds, it returns a counterexample under the form of a sequence of n time-stamped inputs action (1) active (1), . . . , action(n) active(n) that, when replayed on the model and its observer, yields a final output sequence P(1)=1, . . . , P(n-1)=1, P(n)=0 falsified at cycle n. The counterexample demonstrates that MUV violates the property.
978-1-
Our STANCE tool performs a structural analysis of the paths activated by the replay of the counterexample. Before presenting the results of this analysis, we need to clarify what it means for a path to be activated. Indeed, the notion of path activation in a synchronous data-flow model departs from the usual one in control-flow graphs.
A. Path activation in data-flow models
Fundamentally, a synchronous data-flow model is a set of equations defining how input flows are transformed to output flows via some functional operators. At each clock tick, all operators are simultaneously executed to produce values. A digraph graphically represents the equations. The vertices are operators and directed arcs carry data from a producer operator to a consumer one. Given an arc α, we note α(n) the value that A path in the digraph is a non empty sequence of arcs such as, for any pair of consecutive arcs (α i , α i+1 ), the operator entered by α i is the one exited by α i+1 . The path is a potential data propagation channel: the value of the first arc at some cycle n may affect the value of the last arc at some later cycle n + j (j ≥ 0). The time shift depends on the number of delays along the path. To explain it, let us zoom on the RS-latch of the property observer. Figure 2 gives its reference model in terms of basic operators. Path < α 1 , α 3 , α 7 > does not introduce any delay: it connects α 1 (n) to α 7 (n), i.e., the output of the latch may depend on the Set input at the current cycle. Paths of the form < α 1 , α 3 >< α 5 , α 6 , α 3 > k < α 7 >, involving k > 1 traversals of a unit delay, indicate that the output may also depend on the Set input k cycles ago. Note that the number of paths in this model is infinite, due to the temporal loop.
Intuitively, in this data-flow context, the activation of a path means that the data propagation is effective. For example, path p 1 =< α 1 , α 3 , α 7 > is inactive if both the Set and Reset inputs are 1 at the current cycle: the AND operator "stops" the propagation of the Set input to give priority to the Reset. Figure 3 formalizes the path activation conditions for basic operators. It is adapted from [7] (the conditions were originally defined as Boolean Lustre expressions). A condition is built recursively by backward traversal of the path.
According to this definition, a path consisting of a single arc is trivially active at any cycle. For longer paths, the activation condition depends on the traversed logical and temporal operators. A value of 0 is always propagated by an AND operator, while a 1 is propagated only if the other input is also 1. The Switch operator selects its top or bottom input depending on a condition (the middle input): the active paths are via the condition and the selected input. The Unit Delay propagates the value of its input at cycle n − 1, to affect calculation at cycle n. If there is no previous cycle (n = 1), an initialization value is provided. This value is a (constant) input of the operator, and paths originating from it are considered. In Figure 2 , those paths start by the (implicit) arc α init . The activation condition of < α init , α 6 > holds at the first cycle only, while the activation condition of < α 5 , α 6 > holds at any cycle but the first one. Generally speaking, if a path introduces k delays, its activation condition does not hold for the first k cycles. So, even if the number of paths is infinite, the number of paths activated by a finite number of execution cycles is actually finite. Figure 4 exemplifies a concrete execution of the latch Let p =< α 1 , . . . , α k > be a path of length k ≥ 1. The activation condition of p at cycle n ≥ 1, noted AC(p, n), is a predicate defined as follows: This is exactly what STANCE does. The tool identifies subsets of paths active at the last cycle n of the counterexample, and connecting current and past inputs to the falsified sink property P(n). A subset C is a cause of the violation if the input values at the origin of the paths are sufficient to:
• Ensure that all paths in C are active,
Our previous work formalized these sufficient conditions [4] . The formalization refers to variants of the counterexample, that keep the input values determined by the cause but let other inputs receive arbitrary values. Whatever the arbitrary values, the values kept in the cause allow us to control the violation of the property via the identified propagation paths.
B. Analysis of the running example by STANCE
Let us now illustrate the approach on the running example of Figure 1 . The considered MUV is a case study described in [8] . Its active input indicates a temporal interval in which an action is expected, e.g. if active takes the successive values 0110 then action is expected at cycles 2 or 3. If the action never occurred during these cycles, MUV shall issue an alarm. We model this property by having a latch store the occurrence of the action: it is set by action AND active, it is reset by the falling edge of active when the deadline for action has passed. The implication operator expresses that the alarm is required from MUV whenever the end of an active interval is detected, and no corresponding action occurrence was stored before this end event.
The internal design of MUV is shown in Figure 5b (the coloring is added by our STANCE tool, and will be explained shortly). From [8] , we know that this design is flawed. Indeed, when MUV is model-checked, a counterexample is returned. STANCE replays the counterexample and stores the intermediate values of the inputs and outputs of all operators at all cycles. It then performs a structural analysis of the paths activated at the last cycle of the counterexample. It finds two causes. As they are only slightly different, we just present one of them, the one visualized in Figure 5 .
STANCE provides two complementary visual results:
• A list of key events of the cause, including the retained input values and some intermediate values that are important to understand the behavior of the traversed operators; • A colored view of the paths of the cause.
The list of key events ( Figure 5a ) has separate entries for property and MUV events. Similarly, there are separate colored views of the paths via the property observer (not shown in Figure 5 ) and via MUV's internal structure (Figure 5b ). The counterexample of Figure 5 has only two cycles. In the list of the property events, the retained inputs are active(1)=1, active(2)=0, action(1)=0: we have an active interval of one cycle (at cycle 1) with no action. Still, MUV does not raise an alarm at cycle 2. Other intermediate property events concern the behavior of the latch (is not set at cycle 1), the pulse (detects the end of the active interval at cycle 2), and the implication (falsified at cycle 2).
In order to understand why MUV does not raise an alarm, we have to look at MUV events and use the visual support of the colored paths inside MUV. It is striking that the value of the alarm at cycle 2 does not depend on action: the alarm is 0 whether or not there has been an action! The alarm also does not depend on active (1) . So, whatever happens at the first cycle, if active is 0 at cycle 2, the alarm is 0 as well. Clearly, there is an initialization problem at cycle 1. The alarm depends on two initial values, the ones of UD and UD3. In MUV and the property observer, all temporal operators have the initial value 0. These values 0 of UD and UD3 jointly determine the condition of Switch_ActN at cycle 2, so that the constant bottom input is selected.
In [8] , the diagnosis is that the initial value of UD is incorrect. The arc labeled "begin" should carry the value 1 at the beginning of each interval, but the detection of this event does not work at the first cycle. The proposed fix is an initialization of UD to 1. Note that an alternative fix could keep the initial value 0 but swap operators LO1 and UD. Any of these simple fixes would remove the counterexample. But the detailed analysis of the paths may also suggest a more radical change in the design. The initialization structure (wrongly) activated by the counterexample seems unnecessarily convoluted. The constant input 1 entering Switch_ActN is useful only for the initial cycles before a first "begin" is detected. Once a "begin" occurs, the arc labeled "active at least once" is true forever (due to the feedback loop arriving at LO9), which permanently deactivates the path from the constant input. One may wonder whether it is necessary to process the initial inactive cycles differently from other inactive cycles. A simplified design could align their processing, removing the need for the extra constant input and for its selection via a specific switch condition controlled by a feedback loop. We will later present the simplified design we came to.
This small case study illustrates how STANCE facilitates the understanding of a counterexample. But the insight is limited to this counterexample. Should there be any other initialization flaws, or any other types of flaw after the initialization transients 1 , we would not see them. Calling the model-checker with the current version of the model would only return the same counterexample again and again. The only possibility for seeing other ways to violate the property is to eliminate the first one. This may not be the quickest debugging approach, especially if a major revision of the design is considered. Engineers should have the possibility to get more feedback before the revision.
We propose an automated facility for exploring additional counterexamples, exhibiting new causes of violation. It is based on the structural analysis performed for the causes. Intuitively, its principle is to select structural elements that are not activated by the known causes (e.g., the top input of Switch_ActN, or the arc from LO7 to LO5) and to challenge the model-checker to violate the property via paths that traverse these elements.
III. SEARCHING FOR NEW COUNTEREXAMPLES
We present the algorithms we have implemented to search for new counterexamples. The key idea is to synthesize in- 
Outputs : c a u s e s → a r r a y of c a u s e s 3 V a r i a b l e s : i n s t r u m e n t a t i o n s → a l r e a d y i n s t r u m e n t e d a r c s 4 n e w I n s t r u s → new i n t r u m e n t a t i o n s 5 newCauses → c a u s e s from a c o u n t e r e x a m p l e 6 c o u n t e r e x a m p l e → r e t u r n e d c o u n t e r e x a m p l e 7 iModel → i n s t r u m e n t e d model 8 S t a r t 9 c a u s e s = ∅ 10 i n s t r u m e n t a t i o n s = [ ] 11
c o u n t e r e x a m p l e = m o d e l c h e c k i n g ( model ) 12
i f e x i s t s ( c o u n t e r e x a m p l e ) 13 N = l e n g t h ( c o u n t e r e x a m p l e ) 14 modelExec = e x e c u t e ( model , c o u n t e r e x a m p l e ) 15 c a u s e s = f i n d C a u s e s ( modelExec ) 16 i n s t r u m e n t a t i o n s = f i n d I n s t r u m e n t a t i o n s ( modelExec , N, N , ∅ ) 17 f o r each a i n i n s t r u m e n t a t i o n s do 18 iModel = p e r f o r m I n s t r u m e n t a t i o n ( model , a ) 19 c o u n t e r e x a m p l e = m o d e l c h e c k i n g ( iModel ) 20 i f e x i s t s ( c o u n t e r e x a m p l e ) 21 modelExec = e x e c u t e ( model , c o u n t e r e x a m p l e ) 22 newCauses = f i n d C a u s e s ( modelExec ) 23 newCauses = c a u s e s F i l t e r i n g ( newCauses , 
A. Top-Level Algorithm
The top-level algorithm is shown in Figure 6 . Its input is the data structure model, encoding the graph structure of the global model composed of MUV and its property observer (like the global model in Figure 1 ). Its output is a set of causes extracted from all counterexamples found by the search.
The initial steps of the search are described in Lines 9-16 of the algorithm. The global model is model-checked, possibly returning a first counterexample data structure. Replaying this counterexample allows us to produce a modelExec data structure, that supplements the model graph by attaching data to its arcs, i.e., it allows us to store the values carried by the arcs at each execution cycle. The analysis of modelExec returns causes, as explained in Section II. Then, an additional analysis, which is the target of this paper, produces a list of candidate instrumentations to force the model-checker to consider new paths to violations. The instrumentations constrain the values carried by some arcs at some cycles.
Once the search process has been initiated by the processing of the first counterexample, the top-level algorithm iterates over the elements of instrumentations (Lines 17-32 ). An iteration creates an instrumented version of the model: iModel. If iModel allows a counterexample to be found, the counterexample is replayed on the original un-instrumented model (Line 21), to extract its causes (Line 22) and produce further candidate instrumentations (Line 26). Filtering functions are used to eliminate causes already found in other counterexamples (Line 23), as well as instrumentations targeting arcs already covered by previous instrumentations (Line 27). The latter filtering function is critical for ensuring termination of the search. For models including temporal loops, there is potentially an infinite number of paths to violation (iterating n times the temporal loop, n+1 times, etc.). It would be possible to keep producing new causes forever, via counterexamples of increasing length. To avoid endless iteration, we arbitrarily stop the search when the analysis of counterexamples does not allow the instrumentation of new arcs. We now explain the two functions that are at the core of our instrumentation-based approach: findInstrumentations() and performInstrumentation().
B. Finding instrumentations
Function findInstrumentations() uses concrete execution data in modelExec to derive a list of instrumentations forcing the activation of new paths. An instrumentation consists of two fields:
• primary: a set of local constraints for the values carried by the arcs targeted by the instrumentation; • secondary: a set of additional constraints to ensure the propagation of these values and the falsification of the property.
The primary and secondary constraints are encoded by means of three constructors:
• basic(arcId, value, delay): the arc denoted by arcId must carry the given value at the cycle determined by delay; delay is a null or negative number relative to the cycle of the violation, e.g., delay = -1 means one cycle before the violation.
• atLeastOnce(arcId, value, delay): at the cycle determined by delay, the arc denoted by arcId must have carried the given value at least once.
• always(arcId, value, delay): the arc denoted by arcId has always carried the given value until the cycle determined by delay.
Note the relative notion of time introduced by the delay in each case. We do not prescribe a precise length for the new counterexample. The model-checker can build any counterexample, provided its number of cycles N is such that N>delay and the defined constraint holds at cycle N-delay.
To produce the instrumentations, function findInstrumentations() performs a backward traversal of modelExec, both structural (from the sink property to the inputs) and temporal (from the cycle of the violation to previous execution cycles). Each step considers the time-stamped value of an arc that is part of an active violation path. So, the first step considers the arc entering the sink property operator and its value 0 at the last cycle N of the counterexample. All paths of the causes end with this valued arc, which expresses the violation of the property. Function findInstrumentations() identifies the operator at the origin of the arc, i.e., the operator that outputs the value. The operator introduces specific activation conditions for paths that traverse it. As a result, at the level of the operator, some local inputs are currently propagated to produce the output, while 
I n p u t s : s w i t c h → c u r r e n t s w i t c h node 2 N → number o f c y c l e o f t h e c o u n t e r e x a m p l e 3 k → c u r r e n t time−stamp 4 s e c o n d a r y → s e c o n d a r y i n s t r u m e n t a t i o n s 5
Outputs : i n s t r u m e n t a t i o n s → found i n s t r u m e n t a t i o n s 7 To illustrate the approach, let us assume that at some point, we have a recursive call of the form findInstrumentations (arcId, N, k, secondary) i.e., we are currently processing the value carried by the arc arcId at cycle k. The total number of cycles of the counterexample is N. Parameter secondary contains a set of constraints accumulated during the traversal from the root property arc to this arc. It represents a sufficient condition for the value of the arc at cycle k to control the falsification of the property at cycle N (the building of such a sufficient condition is at the core of our analysis for cause extraction). Function findInstrumentations() first identifies the operator at the origin of the arc, say, a Switch operator. It then delegates the processing to a specialized function shown in Figure 7 . Let us assume that the Switch condition is 1 at cycle k. The active paths are thus via the condition and the top input, while the bottom input is ignored. An instrumentation is produced, which forces the output to come from the bottom input. The primary constraints are that the condition is 0 and the bottom input has the value V currently carried by the top input. The secondary constraints are left unchanged. They ensure that the V output of the switch yields the falsification of the property. So, the conjunction of the primary and secondary constraints instructs the model-checker how to violate the property via new paths.
S t a r t 8 i f s w i t c h . c o n d i t i o n . v a l u e ==
Finally, we also search for new paths via the top input and the condition of the Switch. Recursive calls to findInstrumentations() allow us to find alternative ways to produce the same values V and 1 as in the current counterexample. The recursive calls use an updated version of secondary. For example, the 1 I n p u t s : l a t c h → c u r r e n t l a t c h node 2 N → number o f c y c l e o f t h e c o u n t e r e x a m p l e 3 k → c u r r e n t time−stamp 4 s e c o n d a r y → s e c o n d a r y i n s t r u m e n t a t i o n s 5 Outputs : i n s t r u m e n t a t i o n s → found i n s t r u m e n t a t i o n s 7
S t a r t 8 i n s t r u m e n t a t i o n s = [ ] 9
i f o u t p u t ( t i m e ) == 0 10 i f e x i s t s ( l a t c h . r e s e t == 1 a t time t ) 11 i f i n i t V a l u e == 0 12 i n s t r u . p r i m a r y = always ( l a t c h . r e s e t , 0 , −N+k The specialized functions for Boolean operators like Switch produce only basic instrumentations. When it comes to complex temporal operators, more elaborated constraints need to be expressed. For example, Figure 8 presents the algorithm specialized for an RS-latch, producing constraints of the form always and atLeastOnce. In Line 12, the always constraint is produced in the case where the latch output is 0 due to a reset at some previous cycle, and we want to explore whether it is possible to obtain the same result without any reset. So, the reset input is forced to be always 0, in order to activate new paths via the initialization value of the latch. Line 18 uses atLeastOnce in the dual case: the output is currently due to the initialization value, and we want to force an active reset.
i n s t r u m e n t a t i o n s = append ( i n s t r u , f i n d I n s t r u m e n t a t i o n s ( l a t c h . s e t , N, t , s e c o n d a r y ) )
Note that the design of the specialized functions is a matter of compromise, in order to keep a reasonable number of instrumentations. In the RS-latch example, if the current output is 0 due to a reset at cycle t, we do not explore alternative paths via a reset at another cycle t'≤k, nor do we explore the many ways the Set input can be maintained to 0 between cycles t+1 
C. Instrumenting the model
Once candidate instrumentations have been found, we have to effectively perform these instrumentations on the model. This is the role of function performInstrumentation() mentioned in the top-level algorithm. Given a candidate instrumentation I, the function automatically modifies property P into C⇒P, where C is the conjunction of all individual constraints coming from I.primary and I.secondary. In this way, the model-checker is forced to search for counterexamples that satisfy the constraints while falsifying P.
The individual constraints are inserted by means of dedicated Simulink blocks, shown in Figure 9 . For example, a basic constraint is realized by sending the target data to the input port of the basic block. All delays are initialized to 0. In this way, the output of a block is 0 during the first d cycles, yielding the conjunction C of the constraints to be 0 as well. No counterexample of less than d+1 cycles can be returned.
IV. APPLICATION TO THE RUNNING EXAMPLE
Applied to the running example, the automated search gives the following results. The first counterexample found is the one commented in Section II, having two causes. It produces 6 instrumentations summarized in Figure 10 . Some of them come from the exploration of the property observer, while others come from paths traversing MUV. For example, the first instrumentation comes from the property observer. It was decided when analyzing the output of its RS_latch at cycle 1 (i.e., at relative time -1, since the violation is at cycle 2). The counterexample had this output due to the initial value of the latch. The instrumentation forces the output via the activation of the Reset. While iterating over the instrumentations: • Instrumentation 5 finds a counterexample but it is eliminated by the filtering of causes (it has the same cause as the new counterexample from Instrumentation 1).
• Instrumentation 6 returns no counterexample.
So, the search terminates with three causes of violation, two from the initial counterexample, and one from the new one.
The newly found problem is the second flaw reported in [8] . It is quite different from the initialization problem described in Section II. The alarm is not raised if the action comes exactly one cycle too late, i.e., when the activity interval has just ended. In [8] , the proposed fix is to introduce a delay between Switch_Act1 and the subsequent OR operator. We reworked the model based on the insights gained from the causes of the two counterexamples. The result, shown in Figure 11 , is simpler than the original design. Model checking proves that it fulfills the property modeled in Figure 1 . It fixes the detection of the beginning of an activity interval at the first cycle. It homogenizes the processing of initial and subsequent inactivity intervals. It takes care that an action occurring at the first inactive cycle is too late for the previous activity interval. The major revision of the design was greatly facilitated by our ability to gather information on different flaws, as well as by the visualization of the key violation paths and events in each case.
V. CASE STUDY: A FLASHER MANAGER

A. Considered Model
This model, taken from [9] , represents a flasher manager manufactured by Geensoft/Dasssault Systems and is responsible for driving the behavior of the flashing lights of a car. The Simulink model of this device is pictured in Figure 12 . Its features are the following.
a) Direction change:
The driver can indicate a direction change through two Boolean inputs L (for left) and R (for right). The corresponding light (respectively outL and outR) oscillates between on and off states with a 6 time-units period, thus generating a loop of the sequence [111000] as long as the input is true. When the input falls back to false, the lights stop flashing immediately. 
c) Locking and unlocking:
It is also possible to lock and unlock the doors using a 2-buttons RF-key: the locking button (LK) and the unlocking button (ULK). The behavior of the flasher w.r.t both buttons is dependent on the current state of the car (locked or unlocked). As an example, if one presses LK while the car is not already locked, then the lights shall be lit for 10 time-units and then be off for the next 10 time-units, thus producing the sequence [111111111100000000000]. In the initial state, the doors are locked.
Furthermore, the flasher manager has two assumptions on the inputs. The first is that the L and R inputs cannot be triggered at the same time and the second states that the inputs LK, ULK and W inputs are push buttons, meaning the trigger lasts only 1 cycle.
In [9] , the authors considered 4 properties of the flasher manager component. For this paper, we will consider the fourth one. This property states that "the lights should never remain lit infinitely". Being a liveness property, it can not be checked using Simulink's model-checker. We will consider a bounded version of the property, by setting a maximum number of cycles during which the lights shall not remain lit. In [9] , the authors checked the property for bounds of 10, 50, 100 and more (up to 1,600 cycles). They found one counterexample for every bound, but did not discuss the underlying behavior (their focus was on benchmarking model-checking algorithms). So, we know for sure that we will have counterexamples to analyze and we will determine why the property fails. Furthermore, as we are working at a debugging level, we will start with a Figure 13 shows the diagram of the property. It consists of a single Confirmator operator, that outputs 1 only if there has been n+1 consecutive 1 on its input, where n is a parameter (nine in our case, meaning that it detects ten consecutive 1 on its input). For readability purposes, the modeling of the assumptions on the environment is not shown. Also, for simplification, we only check the behavior of the left light, the behavior of the right light being exactly the same.
B. Results
The tool gave us 9 causes each pertaining to a different counterexample. However, after analysis, we managed to categorize those counterexamples into three equivalent classes.
1) First counterexample:
The model-checker found a counterexample where all inputs have been activated at least once (except for R, since we are only checking the left light). However, the colored model gives us a huge hint about what is really happening. Indeed, only the paths containing the Unlocking Activation and Slow Flasher components are involved. This first impression allows us to consider the F, L and W activation as noise, they are thus not relevant for our diagnosis. The associated key events give us even more clues: the falling edge of ULK at time 2 and the rising edge of LK at the same time. The key events also confirm us that F, L and W are not relevant for this counterexample.
What happened is that the activation of ULK unlocked the car and the following LK input locked it back. Now, looking at the behavior of LK in the specification, locking the car when it is in an unlocked state triggers a [111111111100000000000] sequence, hence the violation. This counterexample is not really interesting as it is an artifact of our choice of 10 cycles for checking the property. Fortunately, the next counterexamples are more enlightening.
2) Second counterexample:
The colored model of the counterexample shows a larger number of active areas in the model compared to the previous one. This time, in addition to the ULK and LK activation blocks (Unlocking Activation and Locking Activation, respectively), the areas for W and L are also colored, making them candidates for the responsibility of the violation.
By analyzing the cause, we can determine that the user can lit the light forever, by repeatedly acting on the warning button and on the left direction change. More precisely, in the counterexample, the user activates the W input at cycle 2 (lighting the left light during the cycles 2,3 and 4). Then, at cycle 5, the warning is deactivated and the L input is activated, lighting the lights for another 3 time units (cycles 5, 6 and 7). When the left flasher ends, the W input is activated again keeping the lights on during cycles 8, 9 and 10. Finaly, the L input is activated again to have the light up during cycle 11. At the end, the lights have been on for 10 cycles (from 2 to 11).
The fifth, seventh and eighth counterexamples are variants of this counterexample: user inputs still alternate between warning and left flasher but vary as regards the time to deactivate the warning.
3) Third counterexample:
This counterexample is interesting as it is a mix of the first and the second counterexamples. All parts of the model are colored, so we need to rely on the key events to understand what happened.
We see that the activation of W at cycle 2 is responsible for turning the light on during cycles 2 and 3. On cycle 3, the car is unlocked and is locked back on cycle 4. This part reminds of the first counterexample but now exhibits a new case combining the warning and the lock.
The fourth, sixth and ninth counterexamples are variants of this counterexample, with a different number of warning cycles and different relative times for the unlock before the lock.
C. Conclusion
With these counterexamples, we learned that it is possible to keep a light on with the continuous intervention of the user. We did not find any sequence of inputs that allows the system to remain lit by itself without user interaction.
We introduce an assumption constraining the behavior of the user: there must be at least 4 cycles between consecutive user actions. Setting the maximum lighting limit to 20 (instead of 10) gets rid of the first counterexample, and the added assumption allows the model-checker to end successfully, discovering no new counterexamples.
VI. RELATED WORK
Related work addresses the understanding of counterexamples to facilitate debugging. The interactive visualization of counterexamples is addressed in [10] [11] . [12] and [13] combine visualization and analysis of counterexamples. [12] uses the structure of the property to display information on when the property fails, and with which subconditions. [13] uses the structure of both the model and the property, and annotates counter-examples with proofs explaining the modelchecking result. In software model-checking, [14] and [15] consider neighboring correct and incorrect executions to localize the error in the code. [16] identifies all counterexamples in one pass and uses this information to identify code segments that are likely to cause the violation. [17] combines modelchecking and dynamic analysis of programs. In particular, local coverage information is used to guide the state-space search.
Comparing our work to the above ones, the closest approaches are the ones on software model-checking [14] , [15] [16] [17] . Like them, we analyze the execution of counterexamples in terms of structural locations. However, our work is dedicated to data-flow models. It induces significant differences in how an execution connects to the structure. The execution of an imperative program activates one path, the executed constructs being the ones along the path. In a dataflow model, all constructs are executed at the same time, and a set of temporal propagation paths is activated. So, neighboring executions have to be defined in terms of neighboring sets of paths; temporal propagation conditions (like the ones in our secondary constraints) replace reachability ones.
Outside the model-checking community, structural analysis of data-flow models is performed in testing and debugging work. Ludic [8] is a debugger for Lustre models. It proposes functionalities for the interactive replay of test cases, with the user guiding exploration of the model structure. Structural testing of Lustre models has been addressed by [18] [7] . Particularly, [7] proposed a definition of path activation conditions in data-flow programs, which we re-used in our work.
Finally, work on concolic testing [5] [6] was also a source of inspiration to us. We retained the principle of analyzing concrete execution data to derive constraints for covering new paths. But again, the framework of our approach (data-flow models) is quite different from the one of imperative programs.
VII. CONCLUSION AND PERSPECTIVES
In this paper, we presented a path-based approach for the generation of new counterexamples from a model-checker of data-flow models. It is based on the analysis of the causes of the counterexample, i.e. of the activated paths in the model that were responsible for property violation. Knowing the causes of a counterexample, the approach targets paths that have not yet been involved in a violation. To this end, we developed a set of instrumentation blocks that are used to constrain the value of certain arcs so that new paths are covered. The process is rerun on the instrumented models to find new counterexamples. This approach is implemented in the STANCE tool.
We first considered an academic case study with two known bugs. The approach allowed us to retrieve them, yielding one counterexample for each. We then applied our approach to an industrial-sized case study taken from [9] . We knew from previous work on this case study that the considered property does not hold, but the authors did not analyze why. Our approach gave 9 different counterexamples that we categorized into three classes by analyzing their causes. With this analysis, we understood what went wrong and proposed the addition of an assumption that makes the property successfully checkable.
So far, our approach has focused on analyzing the Boolean and temporal structure of the models, leaving apart the numeric parts. The extraction of the causes assumes that all inputs of a numeric operator (e.g., an arithmetic operator) are propagated, and numeric arcs are not the targets of any instrumentation. Future work will study constraint-based approaches to refine the analysis with respect to the numeric aspects. In particular, approaches extracting subsets of constraints, like Minimal Unsatisfiable Subsets (MUS) or Minimal Correction Sets (MCS) could be considered [19] .
VIII. ACKNOWLEDGEMENTS
This work is funded in part by the French Space and Aeronautic Sciences and Technologies foundation (STAE).
