

# **Test Suite Augmentation for Reconfigurable PLC Software in the Internet of Production**

Marco Grochowsk[i](http://orcid.org/0000-0001-9397-2009)<sup>( $\boxtimes$ </sup>)  $\blacksquare$ [,](http://orcid.org/0000-0001-7348-0146) Marcus Völker $\blacksquare$ , and Stefan Kowalewski $\blacksquare$ 

Embedded Software, RWTH Aachen University, Aachen, Germany {grochowski,voelker,kowalewski}@embedded.rwth-aachen.de

**Abstract.** Regression testing is an established technique used to attest the correctness of reconfigurations to PLC software. After such a reconfiguration, a test suite might not be adequate to ensure the absence of regressions, requiring the derivation of new test cases to uncover potential regressions. This paper presents a combination of state-of-the-art symbolic execution algorithms for test suite augmentation, an indispensable part of regression testing. Test generation is guided towards the changed behavior using a technique known as four-way forking. The old and new PLC software are executed in the same symbolic execution instance to account for the effects of the reconfiguration and increase the chances of generating difference-revealing test cases. The prototypical implementation is evaluated using domain-specific benchmarks such as the PLCopen Safety library and the Pick and Place Unit, exposing the limitations in applicability and effectiveness of the used techniques for safeguarding PLC software subject to frequent reconfigurations as found in cyber-physical production systems.

**Keywords:** Regression testing · Test suite augmentation · Symbolic execution · Programmable logic controllers · Internet of Production

# <span id="page-0-0"></span>**1 Introduction**

Transformability, a property resulting from the flexibility and mechanical reconfigurability of a cyber-physical production system (CPPS), is one of the primary enablers to cope with changing intrinsic and extrinsic demands and is a necessary prerequisite to guarantee the ability to compete with other companies [\[9\]](#page-16-0). An overview of the life cycle and value chain of a CPPS is given in Fig. [1.](#page-1-0) In contrast to a conventional production system, a CPPS is subject to a high degree of reconfigurability during its life cycle. This highly agile manufacturing paradigm leads to an increase in complexity as the insights gained during production turns into data that controls the production process. Due to the heterogeneity and emergent behavior of CPPS, unwanted regressions might accompany those reconfigurations and take their toll on the functional safety and reliability of software components [\[6\]](#page-15-0). In the context of static reconfigurations where the entire CPPS



<span id="page-1-0"></span>**Fig. 1.** Juxtaposition of the life cycle and value chain of cyber-physical production systems (Figure adapted from illustration in [\[19](#page-16-1)]).

is stopped and analyzed during maintenance, short downtimes are crucial, and we argue that lightweight verification techniques such as testing are suitable to assess the CPPS's correctness quickly. Consequently, the goal is to reduce the lead-time after a reconfiguration to the CPPS has occurred by reducing the time it takes to test the reconfigured programmable logic controller's (PLC) software throughout the ramp-up phase during maintenance as depicted in Fig. [1.](#page-1-0)

**Regression Testing.** Regarding the reconfigurations to PLC software, they manifest themselves in the form of the addition of new functionality, the modification of already existing functionality, or the removal of functionality, which most often also requires adaptations to the test suite. As the manual creation of difference-revealing test cases requires enormous effort and expertise, automated techniques are desirable. One prominent set of such automated techniques that tackles test suite maintenance is termed regression testing. Figure [2](#page-2-0) illustrates the process of regression testing and test suite augmentation after a syntactic reconfiguration. Consider the test suite  $T_{all}^P$  for a PLC program P before a reconfiguration with which the reconfigured PLC program  $P'$  should be tested. There are two primary reasons why re-executing the whole test suite is infeasible. The first one is that the test suite might be too large and require too much time while not focusing on the parts of the software affected by the reconfiguration. The other aspect is that the test suite might not even test the changed behavior of P . In this sense, test suite augmentation is necessary and an important complementary technique to traditional regression testing techniques [\[21,](#page-16-2)[23\]](#page-17-0). Dealing with reconfigurations to the PLC software and its effect on the test suite is a two-step approach during test suite maintenance. First, one has to assess if the test suite  $T_{all}^{\overline{P}}$  is still *adequate* enough for testing P'. Standard measures for adequacy are whether the test suite is homogeneous with regards to the program paths, for instance, line or branch coverage. Nonetheless, one has to keep in mind that coverage alone does not quantify the capability of a test suite to reveal regressions. If the test suite is not homogeneous with regards to the failure [\[20\]](#page-16-3), i.e., it structurally covers the reconfigured program path but does not propagate a divergence to the output, it will not reveal the regression after a faulty syntactic



<span id="page-2-0"></span>**Fig. 2.** Application of regression testing techniques and test suite augmentation after a syntactic reconfiguration.

reconfiguration. Second, the reconfigurations in  $P'$  need to be identified, and the test case generation algorithm has to be guided to cover the potentially reconfigured behavior. As regressions are only observable for inputs that expose a behavioral difference, we use a concept coined as four-way forking [\[10\]](#page-16-4) to guide the test case generation into parts of the software affected by a reconfiguration. As the identification of the reconfiguration is a challenging problem, we resort to manual software annotations to explicitly denote the reconfigured parts from one version to another.

**Syntactic Reconfiguration.** The syntactic reconfiguration mentioned in Fig. [2](#page-2-0) follows the concept presented in  $[10]$  $[10]$ , where a change (old,new) macro was used to characterize the effect of the reconfiguration. The first argument of this macro represents the expression from the PLC software before the reconfiguration, and the second argument represents the expression of the PLC software after the reconfiguration. As a result, the manifestation of reconfigurations to PLC software stated earlier, i.e., the addition of new functionality, e.g., adding an extra assignment  $x := \texttt{change}(x, 1);$ , the modification of already existing functionality, e.g., changing the right-hand side of an assignment  $x := y + \text{change}(1, 2)$ ;, or the deletion of functionality, e.g. removal of straightline code if(change(true, false))...code... can be expressed succinctly with the change (old,new) macro. This way of annotating the expressions of reconfigured parts of the software has a significant benefit as it keeps the correspondence between both versions intact and was therefore chosen for analyzing the semantic effects of the implication introduced by the reconfigurations.

#### **1.1 Limitations and Contributions**

A premise resulting from the introduction is the existence of syntactically changeannotated PLC programs given as input to our framework. To further narrow the scope of this contribution, the peculiarities of PLCs have to be considered. A PLC is subject to cyclic execution resulting in non-termination. Still, every execution through one cycle terminates and hence can be analyzed. The programming languages for PLCs forbid recursive calls, i.e., the call-flow graph is acyclic [\[8](#page-16-5)]. Furthermore, our framework does not support the use of arrays or pointers yet. Nevertheless, statically allocated memory can be modeled by flattening the arrays.While the prototypical framework is able to analyze loops other than the naturally occurring execution cycle of the PLC program, these loops are not explicitly handled and analysis might be intractable. As some of the benchmarks use the timer capabilities of the IEC 61131-3 standard [\[8](#page-16-5)], we use an over-approximating representation of timers from [\[1\]](#page-15-1), which non-deterministically models the internal decision variable measuring the passing of time. Last but not least, control tasks are usually distributed in the context of Industry 4.0, yet most often still coordinated centrally. Instead of having a single PLC that controls the various actuators in the CPPS, multiple PLCs exist, one for each control task and one overarching, coordinating PLC. Despite that, we model the distributed control task as one, compositional, *classic* PLC program, in which the other control tasks are incorporated as function blocks and executed on one single PLC controller (cf. Sect. [4\)](#page-11-0). This neglects the influences of different times and latencies introduced due to the communication between each controlling PLC. We assume that the sequential modeling using a single PLC is a feasible abstraction of several distributed PLCs running in parallel, realizing the same control task, because the business logic is implemented by a single, coordinating PLC, which processes the messages of the other distributed PLCs sequentially in all circumstances. To this end,

- we improve the scalability of an existing Dynamic Symbolic Execution (DSE) algorithm for PLC software,
- we evaluate the feasibility of DSE and the concept of four-way forking for test suite augmentation of reconfigured PLC software on benchmarks of varying difficulty and compare it to previous results.

# **2 Related Work**

Symbolic execution is one of the primary techniques for software testing and resulted in the development of numerous language-agnostic analysis tools in the past [\[3\]](#page-15-2). Previous work has investigated the applicability of DSE in test suite generation for PLC software [\[4\]](#page-15-3). The results were promising but have not been applied to tackle the problem of test suite augmentation after a reconfiguration. In contrast to [\[4\]](#page-15-3), the concolic and compositional DART algorithm, also known as SMART [\[5](#page-15-4)], explores the program execution tree depth-first on a per path basis allowing for the use of summaries. However, we currently refrain from summarization due to our conflicting merging strategy. An approach that aids regression testing with static change-impact analysis is called directed incremental SE (DiSE) [\[22\]](#page-16-6). The rationale behind this is that static analysis avoids the problems of undecidability of whether there exists an input that is differencerevealing against the reconfigured program by over-approximating the semantic properties using syntactic dependencies such as control- and data dependencies.

The results from the static analysis are used to guide symbolic execution by exploring only paths that can reach parts of the software affected by the reconfiguration. This approach, however, has two severe disadvantages. We argue that these *slices* give only conservative estimates and are often too imprecise, reducing opportunities for information reuse from the prior analysis of the reconfigured PLC software. Furthermore, DiSE only explores one execution path through the impacted parts of the software, and besides reachability, there is no guidance in the direction of real divergences. This lead us to the choice of Shadow Symbolic Execution (SSE) [\[10](#page-16-4)] for test suite augmentation. SSE uses a seeded exploration with a test case that touches the presumable *patch*, or in our terminology, the reconfiguration. The novelty of SSE is that it executes the old (presumed buggy) and new (presumed patched) program within the same SE instance. Therefore, it allows the algorithm not to re-execute potentially expensive path prefixes, which provides several opportunities to prune and prioritize paths and simplify constraints. Despite this, the reconfigurations are touched by a test case that dictates the context in which the potential reconfiguration is reached and hence limits the generalization. Furthermore, both programs need to be merged into a change-annotated, unified version.

**Verification and Testing in the PLC Domain.** Regarding the safeguarding of reconfigurations in the PLC domain several techniques on various levels have emerged in the past years. TESTIAS  $[24]$  $[24]$  is a tool for model-based verification of reconfigurations to distributed automation systems. It works on a higher level than PLC software, i.e., trying to prove the correctness of a reconfiguration affecting the functional perspective of services in a CPPS. Prioritization for regression testing of reconfigured PLC software with regards to system tests was evaluated in [\[17](#page-16-7)]. It optimizes the regression testing process of CPPS after a reconfiguration, however, it is unable to generate difference-revealing test cases. Another interesting approach poses the modular regression verification of the VerifAps library which was successfully applied to the Pick and Place Unit (PPU) case study in [\[18](#page-16-8)]. Modular regression verification requires the specification of relational regression verification contracts allowing for the decomposition of the verification task resulting in efficient solving, yet being far from a pushbutton technology.

### <span id="page-4-0"></span>**3 Methodology**

An overview of our prototypical test suite augmentation (TSA) framework is given in Fig. [3](#page-5-0) and explained throughout this section. TSA can be considered as a development time technique, in which the developer manually annotates the desired changes and is able to assess their implications on the observable behavior of the PLC software. The input to the program analysis framework is a manually change-annotated PLC program in structured text (ST), one of the five IEC 61131 programming languages [\[8\]](#page-16-5), using the change(old,new) annotation macro introduced in Sect. [1.](#page-0-0) Before going in-depth with the core TSA algorithm, we briefly describe our intermediate representation of the PLC software.



<span id="page-5-0"></span>**Fig. 3.** Overview of the prototypical TSA framework.

### <span id="page-5-1"></span>**3.1 Intermediate Representation**

A PLC program can consist of several program organization units (POUs), which provide an interface definition of the input, local, and output variables, and a body containing the actual instructions that operate on this interface. The IEC 61131 standard [\[8](#page-16-5)] distinguishes between three types of POUs, namely functions, function blocks, and programs. A program represents the main entry, whereas function blocks and functions represent stateful and stateless procedures, respectively. At cycle entry, new input values are read from the environment and written to the input variables. During the execution of the cycle, the program operates on a copy of these input variables and internal state variables. The state variables also comprise output variables written to the PLC's output at the cycle exit. While new values are assigned to input variables in each cycle, the internal state variables retain their values. During the parsing and compiling of the input program, function blocks are lowered to *regular* procedures operating on references of their variables. As a result, parameterized calls to function blocks are modeled as parameterless calls preceded and succeeded by a sequence of input and output assignments in the respective caller, which do not modify the state explicitly but rather transfer the flow of control between procedures. For this purpose, we have chosen a goto-based intermediate representation (IR) to represent a subset of the ST language [\[8](#page-16-5)] in form of a so-called control-flow graph (CFG) [\[2](#page-15-5)]. We model the PLC program as a pair  $P = (G, \mathcal{G})$ , where  $G \in \mathcal{G}$ is the CFG of the program POU, and  $\mathcal G$  is a set of CFGs representing nested function blocks occurring in the program. The instructions supported by this IR are defined over variables  $x \in \mathbf{X}$ , Boolean and arithmetic expressions e as usual

 $I ::= \mathbf{assign}(x, e) | \mathbf{ite}(e, \mathbf{goto}\ b_{\ell_1}, \mathbf{goto}\ b_{\ell_2}) | \mathbf{call}\ G' | \mathbf{return} | \mathbf{cycle} |.$ 

Unlike in typical goto-based IRs, we introduced a **cycle** instruction, explicitly denoting the end of the execution cycle. Given the terminology, we will dive into the baseline symbolic execution framework used for generating the test cases which is reused during the application of SSE.

#### **3.2 Bounded Symbolic Execution**

Our implementation of the Bounded Symbolic Execution (BSE) for TSA is composed of three components: an execution context, an executor, and an exploration strategy. An execution context  $q = (c, \ell, f, \rho, \sigma, \pi)$  consists of a cycle c, a label  $\ell$ referring to a vertex  $b_{\ell}$  of a CFG  $G$ , a frame stack  $f$ , a concrete store  $\rho$ , which associates variables with concrete values, a symbolic store  $\sigma$ , which associates variables with symbolic values, and a path constraint  $\pi$ . The frame stack f holds triples  $(G_{\text{callee}}, \text{scope}, \ell_{\text{caller}})$ , where  $G_{\text{callee}}$  denotes the CFG of the callee, *scope* is the scope in which the call occurred, and  $\ell_{\mathit{caller}}$  denotes the intra-procedural label of the caller at which the execution should resume after returning from the callee. The BSE algorithm is given in Algorithm [1](#page-7-0) and explained in the following. It is also commonly known as compositional SE in literature [\[3](#page-15-2)] augmented with parameterizable local and global termination criterias.

**Exploration Strategy.** We decided for a cycle-based, depth-first exploration strategy similar to [\[4](#page-15-3)] with parameterizable timeout, coverage, and cycle bounds. As the cyclic execution of PLC programs significantly increases the computation time of symbolic execution, we adjusted the termination criteria in line 2 to consider a configurable cycle exploration bound. The priority queue  $\mathcal Q$  is sorted heuristically by prioritizing execution contexts with a lower cycle count, resulting in the exploration of all feasible execution paths through one execution cycle before continuing with the next cycle. Furthermore, candidate execution contexts with a deeper path length and a concretely executable store are prioritized over execution contexts with a shallower path length. This enables the depth-first exploration to simulate a breadth-first exploration through one cycle and generates concise test cases with no unnecessary executed cycles. When encountering the end of the cycle during execution (cf. line 25), the cycle counter is increased and new concrete input valuations and fresh symbolic variables are derived.

**Assignments, Branches and Calls.** The semantic effects of the instructions on the respective stores are captured via an evaluation function eval. For an assignment **assign** $(x, e)$ , the concrete and symbolic store are updated via  $\rho \leftarrow \rho[x \mapsto$ eval<sub> $\varrho$ </sub>(e)] and  $\varsigma[x \mapsto \text{eval}_{\varphi}(e)]$ , respectively, as illustrated in line 10. The bracket notation [] denotes the usual replacement for the specified variable in the store. Whenever an  $\textbf{ite}(e, \textbf{goto} \ell_1, \textbf{goto} \ell_2)$  instruction is encountered, the path constraint is updated symbolically depending on the result of the branch expressions concrete evaluation (cf. line 12). In case the expression evaluates to true, execution is continued in the positive branch and a test case is derived if this label is yet uncovered. We also check if the other path is feasible under the current path constraint and fork the execution context with the concrete valuation of the model (cf. lines 15–19). As mentioned in the beginning of Sect. [3.1,](#page-5-1) call and return effects are lowered to input and output assignments during compilation. Therefore, the **call** and **return** instruction modify the frame stack and update the control-flow accordingly.

```
Algorithm 1: Bounded Symbolic Execution
```

```
Input : Program P = (G, \mathcal{G}), CFG G = (\mathbf{X}, \mathbf{X}_{in}, (B, E), b_{\ell_e}, b_{\ell_x})Output : Test Suite T
   \mathcal{Q} \leftarrow \{ (0, \ell_e, \emptyset, \rho_{\ell_e}, \sigma_{\ell_e}, true) \}; \mathcal{M} \leftarrow \emptyset2 while (Q \neq \emptyset \vee \mathcal{M} \neq \emptyset) ∧ ¬terminationCriteriaMet do
   3 if Q = \emptyset then Q.push(merge(M))<br>4 q \leftarrow (c, \ell, f, \rho, \sigma, \pi) \leftarrow Q.pop()
   4 q \leftarrow (c, \ell, f, \rho, \sigma, \pi) \leftarrow \mathcal{Q}.\texttt{pop}()5 if reachedMergePoint(q) then
  \begin{array}{c|c} \mathbf{6} & \mathbf{6} \\ \mathbf{7} & \mathbf{else} \end{array}7 else
   \textbf{8} \quad | \quad | \quad \textbf{switch} \; \texttt{instrument}(\ell) \; \textbf{do}9 case assign(x, e) do
\begin{array}{|c|c|c|c|c|}\hline \textbf{10} & & \end{array} \begin{array}{|c|c|c|c|c|}\hline \textbf{10} & & \end{array} \begin{array}{|c|c|c|c|c|}\hline \textbf{10} & & \end{array} \begin{array}{|c|c|c|c|c|c|}\hline \textbf{1}_{\mathcal{D}}(c) & & \textbf{0}_{\mathcal{D}}(c) & \textbf{0}_{\mathcal{D}}(c) & \textbf{0}_{\mathcal{D}}(c) & \textbf{0}_{\mathcal{D}}(c) & \textbf{0}_{\mathcal{D}}(c) & \textbf\begin{array}{|c|c|c|}\n\hline\n\textbf{11} & \textbf{\textit{}} & \textbf{\textit{}} & \textbf{\textit{}}\n\end{array}\n\textbf{case} \textbf{ite}(\textit{e}, \textbf{goto} \textit{ \ell }_{1}, \textbf{goto} \textit{ \ell }_{2})\textbf{ do}12 \vert \vert \vert \vert if eval<sub>\rho</sub>(e) then
\begin{array}{|c|c|c|c|c|}\n\hline\n\textbf{13} & & q_1 \leftarrow (c, \ell_1, f, \rho_1, \sigma_1, \pi \wedge \texttt{eval}_{\sigma}(e)); \textbf{Q}.\texttt{push}(q_1) \\\hline\n\end{array}\begin{array}{|c|c|c|c|}\n\hline\n\textbf{14} & & \textbf{if } \neg \texttt{covered}(\ell_1) \textbf{ then } T.\texttt{deriveTestCase}(q_1) \\\hline\n\end{array}15 if \mathbf{tryFork}(\pi \wedge \text{eval}_{\sigma}(\neg e)) then<br>
16 i \theta_2 \leftarrow \text{model}(\pi \wedge \text{eval}_{\sigma}(\neg e)))16 \rho_2 \leftarrow \text{model}(\pi \wedge \text{eval}_{\sigma}(\neg e))<br>
17 q_2 \leftarrow (c, \ell_2, f, \rho_2, \sigma, \pi \wedge \text{eval}_{\sigma}(\neg e))\begin{array}{|c|c|c|c|c|}\n \hline\n \textbf{17} & & q_2 \leftarrow (c, \ell_2, f, \rho_2, \sigma, \pi \wedge \texttt{eval}_{\sigma}(\neg e)); \mathcal{Q}.\texttt{push}(q_2) \hline \end{array}\begin{array}{|c|c|c|c|c|}\hline \textbf{18} & & \textbf{19} & \textbf{18} & \textbf{19} & \textbf{19} & \textbf{18} & \textbf{19} &19 end
20 else else // analogous, omitted for brevity
\begin{array}{|c|c|c|}\n\hline\n\text{21} & \text{} & \text{case call } G' \text{ do} \end{array}\begin{array}{|c|c|c|} \hline \textbf{22} & & f.\texttt{push}(G',\texttt{getScope}(G),\ell_{G_r}); \textbf{\textit{Q}.push}\big((c,\ell_{G'_e},f,\rho,\sigma,\pi)\big) \hline \end{array}23 case return do
24 \begin{bmatrix} \phantom{i} \end{bmatrix} \phantom{i} \begin{bmatrix} \phantom{i} \end{b25 case cycle do
26
                                                   Q.\text{push}((c + 1, \ell_e, f, \rho[x \in \mathbf{X}_{in} \mid x \mapsto \texttt{random}(),\sigma[x \in \mathbf{X}_{in} \mid x \mapsto x_{\text{fresh}}], \pi)27 end
28 end
29 end
30 return T
```
**Merge Strategy.** Execution contexts are merged at all possible points where the control-flow joins with respect to realizable paths as opposed to merging at the cycle end as in [\[4\]](#page-15-3). During execution, we check whether the current context reached an interprocedural realizable merge point (cf. line 5) and add it to the merge queue M for further processing.

**Unreachable Branches.** The detection of unreachable branches is an essential task to avoid the encoding of infeasible paths when applying symbolic execution. As our static analysis (SA) is currently not capable of abstract interpretation, we leveraged the algorithms from  $C_{\rm RAB}^1$  $C_{\rm RAB}^1$  to build a value set analysis calculating the

<span id="page-7-1"></span><sup>1</sup> [https://github.com/seahorn/crab.](https://github.com/seahorn/crab)

possible values for each variable at each label of our CFGs. Using this information, we can statically deduce whether a branch is reachable or not.While being a powerful tool it is apparent that the SA of CRAB is not tailored to the domain of PLC software. To express our IR in a form such that CRAB is able to analyze it, it passes several code transformation pipelines including basic block encoding, three-address code, call-transformation and static single assignment which severely bloats up the CFG representation. In order to get precise information the expensive boxes domain was chosen [\[7\]](#page-15-6). The boxes domain is sensitive to the number of "splits" on each variable which come, among other things, from joins and Boolean operations. Unfortunately, the benchmarks in Sect. [4](#page-11-0) "split" a lot due to the cyclic dependency between variables and the state-machine like behavior. Therefore, to still be able to reuse at least some information from the SA, we decided for a trade-off between precision and run time by tuning the behavior of the boxes domain to convexify after a certain amount of disjunctions resulting in imprecise but still usable results.

#### **3.3 Shadow Symbolic Execution**

Intuitively, two things are needed for TSA after a reconfiguration: (1) the test cases must reach potentially affected areas along different, relevant paths (specific chains of data- and control-dependencies), and (2) test cases must account for the state of the PLC software and the effects of the reconfigurations, i.e., be difference-revealing. An interesting research question in this context is whether the concept of four-way forking stemming from the SSE [\[10\]](#page-16-4) algorithm is applicable to the PLC domain using the change(old, new) macros (cf. Sect. [1\)](#page-0-0) to apply TSA for reconfigurable PLC software. In general, it can be intractable, because outputs are potentially difference-revealing after  $k$  cycles (depending on the internal state) and hence the analysis runs out of memory before the difference is reached. In general, deriving difference-revealing test cases in the style of SSE [\[10](#page-16-4)] is a two-step application of SE algorithms (cf. Fig. [3\)](#page-5-0) and is presented in detail in Algorithm [2.](#page-8-0) In line 1 of Algorithm [2](#page-8-0) the test suite of the version before the reconfiguration is reused and executed on the change-annotated PLC program to determine which test cases "touch" the change. Prior to execution, in case the interface has changed due to the

<span id="page-8-0"></span>

reconfiguration, the test case does not contain valuations for all variables. Therefore, we augment the test case with additional valuations using the 0-default initialization for *BOOL* and *INT* as defined in IEC61131-3, *false* and 0, respectively. Each executed test case is further augmented with additional information such as the execution history and state valuations reaching the end of the cycles of the old program version. As a test case can "touch" multiple change-annotated labels, we consider only the test cases that cover as much information as possible with regards to the respective change-annotated label. This reduces the amount of test cases needed for consideration in the first phase without losing expressiveness, as test cases spanning along multiple cycles with the same prefix are prioritized. When functionality is added depending on newly introduced input variables, the prior test suite is unable to cover these labels, hence we keep track of labels that were change-annotated but not "touched" by any test case.

**Finding Divergent Contexts.** Before continuing with the explanation of Algorithm [2,](#page-8-0) we present how divergent contexts are found during symbolic execution. Algorithm [3](#page-10-0) uses the concept of four-way forking to determine whether the execution of a test case leads to potential divergent behavior or not. It is driven by the concrete input valuations of the corresponding test case (cf. line 1) and the augmented BSE is concolically executed on a per cycle basis using a single execution context, hence no merging. In general, the algorithm is similar to the one presented in Algorithm [1.](#page-7-0) We adapted the handling of branches to support the four-way forking and introduced additional data structures for storing the shadow expressions in the context, here hidden behind the concrete and symbolic store. As change annotations may occur in any instruction (or expression) we use the notion of symbolic change shadows and check whether such a change shadow influences the behavior of the current execution path. In case a branch is encountered during the concolic execution of the test case, we recursively check if the expression contains a symbolic change *shadow* (cf. line 7). If the current branch expression contains no shadow expression, we continue the execution as illustrated in Algorithm [1](#page-7-0) in the lines 12– 20. In case the branch expression contains a shadow expression, it might lead to divergent behavior. In order to check whether the current test case takes different paths in the old and the new version of the code, we first evaluate it under the concrete store of the divergent context resolving all shadow expressions (cf. line 8). If the valuations of the expression in the old and the new context do not coincide, the test case exposes truly divergent behavior which might trigger difference-revealing outputs. At this point, the execution stops and the divergent context is added to the queue to be explored in the second phase. If the valuations are equal, there still might be potential divergent behavior. First, we encode the expression using the old and the new symbolic valuations and then check in lines 14–17 whether potential divergent behavior exists. For this purpose, we explore subsequently whether there exist concrete valuations that may diverge and derive a test case as a witness. The forked divergent context is added to the divergent context queue and the execution continues with either following the true or the false branch trying to propagate the execution context to a deeper nested potentially divergent

```
Algorithm 3: findDivergentContexts – BSE with four-way forking [10]
```

```
\textbf{Input} \qquad \textbf{: } \text{CFG } G = (\textbf{X}, \textbf{X}_{in}, (B, E), b_{\ell_e}, b_{\ell_x}), \text{ Test Case } tOutput : Divergent Contexts Q_{divergent}1 foreach c_t \in t do<br>2 d a \leftarrow (c \mid t \mid a^{ct})2 q \leftarrow (c, \ell, f, \rho_{input}^{ct}, \sigma, \pi)3 while c = c_t do
  \textbf{4} \quad | \quad \textbf{switch} \text{ instructionAt}(\ell) \text{ do}5 // other cases analogous to BSE, omitted for brevity
  \textbf{6} \quad | \quad | \quad | \quad \textbf{case}~\textbf{ite}(e, \textbf{goto}~\ell_1, \textbf{goto}~\ell_2)~\textbf{do}7 i if containsShadowExpression(eval<sub>\sigma</sub>(e)) then
  \begin{array}{|c|c|c|c|}\n\hline\n8 & 1 & 0 \ \hline\n\end{array} \begin{array}{|c|c|c|c|}\n\hline\n\end{array} \begin{array}{c}\n\hline\n\end{array} \begin{array}{c}\n\hline\n\end{array} \begin{array}{c}\n\hline\n\end{array} \begin{array}{c}\n\hline\n\end{array} \begin{array}{c}\n\hline\n\end{array} \begin{array}{c}\n\hline\n\end{array} \begin{array}{c}\n\hline\n\end{array} \begin{array}{c}\n\hline\n\end{array} \begin{array}{c}\n\hline\n\end{array} \begin{array}{9 if v_{old} \neq v_{new} then // divergent behavior
\begin{array}{|c|c|c|c|c|}\n\hline\n\textbf{10} & & & \textbf{10} & \textbf{11} & \textbf{2}_\textit{divergent}.\texttt{push}((q, t))\n\hline\n\end{array}11 return Q_{divergent}<br>
12 return Q_{divergent}12 else // potential divergent behavior
13 ( \varphi_{\text{old}}, \varphi_{\text{new}}) \leftarrow \text{eval}_{\sigma}^{\text{shadow}}(e)14 if tryDivergentFork(π ∧ ¬\varphi_{old} ∧ \varphi_{new}) then<br>15 i \theta_{new} Odiscress DUSh((Gerted deriveTestCase(αε
\begin{array}{|c|c|c|c|c|}\n \hline\n \text{15} & & \text{if } \mathcal{Q}_{\textit{divergent}}.\text{push}\big((q_{\textit{forked}},\texttt{deriveTestCase}(q_{\textit{forked}}))\big) \\\hline \end{array}16 end
17 if tryDivergentFork(π ∧ \varphi_{old} ∧ ¬\varphi_{new}) then<br>
18 i \theta_{dimerant} Dish((Gested, deriveTestCase(a
\begin{array}{|c|c|c|c|c|}\hline \textbf{18} & & \textbf{18} & \textbf{19} & \textbf{18} \ \hline \end{array}19 end
20 i i if v_{\text{old}} then
\begin{array}{|c|c|c|c|c|}\hline \textbf{21} & & & \ \hline & & & \ \hline & & & & \ \hline & & & & \ \hline & & & & q & \leftarrow & \ (c, \ell_1, f, \rho, \sigma, \pi \wedge \varphi_\textit{\scriptsize{old}} \wedge \varphi_\textit{\scriptsize{new}}); \hline \end{array}22 else
\begin{array}{|c|c|c|c|c|}\hline \textbf{23} & & & \ \hline \end{array} \begin{array}{|c|c|c|c|c|}\hline \textbf{24} & & \ \hline \end{array} \begin{array}{|c|c|c|c|c|}\hline \textbf{25} & & \ \hline \end{array} \begin{array}{|c|c|c|c|c|}\hline \textbf{26} & & \ \hline \end{array} \begin{array}{|c|c|c|c|c|}\hline \textbf{27} & & \ \hline \end{array} \begin{array}{|c|c|c|c|c|}\hline \textbf{28} & & \ \hline \end24 end
25 end
26 else // analogous to BSE, omitted for brevity
27 end
28 end
29 end
30 end
31 return Qdivergent
```
<span id="page-10-0"></span>context. On termination, i.e., either when a divergent context is found or when all the concrete input valuations for each cycle of this test case have been executed, the algorithm returns the set of divergent contexts and continues with the next test case.

**Propagating Divergent Contexts.** The second phase of Algorithm [2](#page-8-0) performs a seeded BSE (cf. Algorithm [1\)](#page-7-0) for each found divergent context in the first phase. The divergent context and test case passed as parameters in line 6 represent either a diverging concrete execution or were generated because of a potential, possible divergence at the four-way fork in the first phase. This phase runs until the termination criteria is met and tries to generate as many test cases as possible. These test cases cover paths originating from a divergence and hence may expose differences in the outputs between the old and the new version of the reconfigured PLC program. In line 8 of Algorithm [2](#page-8-0) the derived divergent test cases are checked for output differences. The execution of modified instructions does not mean that they are necessarily difference-revealing because the subdomains do not need to be homogeneous with regards to the failure [\[20\]](#page-16-3). Hence to determine whether a test case exposes an externally observable difference, the outputs on the test case in the new version are compared to the outputs on the test case in the old version. If the outputs differ on a per cycle basis, the test case is added to the set of difference-revealing test cases and requires further examination by the developer.

## <span id="page-11-0"></span>**4 Evaluation**

The evaluation was conducted on an Intel(R) Core(TM) i5-6600K CPU  $@$ 3.50 GHz desktop with 16 GB of RAM running openSUSE Leap 15.3. For SMTsolving, we utilized the high-performance automated theorem prover Z3 by Microsoft [\[13](#page-16-9)]. The benchmarks evaluated with ARCADE.PLC were also run with the same evaluation setup. The code of our contribution and the corresponding benchmarks are available for download at [https://github.com/embedded](https://github.com/embedded-software-laboratory/TSA-FMICS22)[software-laboratory/TSA-FMICS22.](https://github.com/embedded-software-laboratory/TSA-FMICS22)

In the following, we first present the achieved performance improvements for the BSE as our TSA implementation heavily relies on it before presenting the results of the TSA algorithm on a few selected benchmarks.

**PLCopen Safety Suite.** The benchmark consists of a set of safety-related PLC programs provided by the PLCopen organization [\[15](#page-16-10)]. The results are listed in Table [1](#page-12-0) and show for each evaluated function block the lines of code (LOC), the coverage values as well as the runtimes of the implementation of a merge-based test generation algorithm in Arcade.PLC [\[4\]](#page-15-3) in comparison to the results of our contribution. Because both tools use different IRs, the number of reachable branches is omitted. The timeout (TO) was set to 10 min. For the detection of unreachable branches, Arcade.PLC uses a values-set analysis, however, we did not add the time to the results. Instead, we ran both programs with this additional pre-computed information to only focus on the performance of the DSE algorithms. The  $SA_{manual}$  refers to the use of CRAB and manual annotation for truly unreachable branches which were over-approximated due to the convexification of the disjunctions (cf. Sect. [3\)](#page-4-0). As far as the function blocks are concerned, both approaches perform equally well. As all blocks follow the same general structure, the LOC can be seen as a reference for giving a rough estimate on what one would expect time wise from the analysis. A significant difference between both approaches is the amount of test cases generated. While ARCADE.PLC generates concise test cases for every branch, our contribution tries to avoid redundancies due to shorter test cases being included in longer test cases, hence generating less test cases overall. This is neither a benefit nor a disadvantage and could be obtained

by a static postprocessing on the test suite generated by ARCADE.PLC. Do note that Arcade.PLC does not dump any test cases in case it runs into a TO due to a technical limitation. The programs on the bottom half are bigger in the sense that they are composed of multiple function blocks from the top with additional logic and were analyzed without manual SA annotation. As more and more calling contexts are available it becomes apparent that delaying the merging until the end of the cycle performs way worse than merging on all realizable paths when the opportunity emerges. Most notably, the performance degenerates on blocks which make heavy use of timer and edge trigger function blocks because only specific paths can reach deeper behavior.



<span id="page-12-0"></span>**Table 1.** Comparison of branch coverage and runtimes for the test generation of the PLCopen Safety library, ordered alphabetically.

Pick and Place Unit (PPU). The benchmark consists of a total of 15 scenar-ios for the PPU of an open-source bench-scale manufacturing system<sup>[2](#page-12-1)</sup>. While it is

<span id="page-12-1"></span><sup>2</sup> [https://www.mw.tum.de/ais/forschung/demonstratoren/ppu/.](https://www.mw.tum.de/ais/forschung/demonstratoren/ppu/)

limited in size and complexity, this trade-off between problem complexity and evaluation effort does not harm the expressiveness of the benchmark. In this evaluation, we focused on the first four scenarios and translated them from their PLCopen XML representation to ST using the VERIFAPS library<sup>[3](#page-13-0)</sup>. The Scenario  $\Omega$  consists of a stack, crane and a ramp of which the latter is only mechanical. The reconfiguration Scenario  ${0 \rightarrow 1}$  aims to increase the ramp's capacity. This reconfiguration does not affect the software as the ramp is a purely mechanical component. As a response to changing customer requirements, the reconfiguration Scenario  $\{0 \rightarrow 2\}$  enables the PPU to handle both plastic and metallic workpieces. For this purpose, an induction sensor is introduced which changes the output behavior of the stack component. The behavior of the crane is untouched. The third reconfiguration Scenario  $\{2 \rightarrow 3\}$  introduces the stamping functionality of metallic workpieces. This impacts the behavior of the crane as workpieces need to be stamped before being transported to the ramp. The results of the test suite generation using BSE without SA results are shown in Table [2.](#page-13-1) The PPU has more complex behavior in comparison to the PLCopen safety suite, which is also reflected in the required time/termination criteria for the test case generation. A comparison with Arcade.PLC was omitted as it was not able to analyze the benchmarks.

<span id="page-13-1"></span>

| <b>PPU</b>                                        |       | Contribution |     |         |    |  |  |  |
|---------------------------------------------------|-------|--------------|-----|---------|----|--|--|--|
| Scenario LOC cov. [%] $T [\#]$ time [s] cycle [#] |       |              |     |         |    |  |  |  |
| Scenario O                                        | - 412 | 88.97        | 45  | 169.82  | 25 |  |  |  |
| Scenario 1                                        | - 412 | 88.97        | 45  | 170.12  | 25 |  |  |  |
| Scenario 2                                        | -459  | 89.61        | 55. | 274.19  | 25 |  |  |  |
| Scenario 3                                        | 768   | 91.67        | 102 | 1198.08 | 25 |  |  |  |

**Table 2.** Results of the test suite generation using BSE for selected PPU scenarios.

Table [3](#page-13-2) shows the results of the TSA for the manually change-annotated reconfigured PLC programs.

<span id="page-13-2"></span>**Table 3.** Results of the TSA using Algorithm [2](#page-8-0) for selected reconfiguration scenarios of the PPU.

| <b>PPU</b><br>Evolution                               | $\begin{array}{c} \ell_{ca} \ [\#]/ \ \ell_u \ [\#] \end{array}$ |              | Phase 1                       |       | Phase 2                             |         |                |
|-------------------------------------------------------|------------------------------------------------------------------|--------------|-------------------------------|-------|-------------------------------------|---------|----------------|
|                                                       |                                                                  | $T_{ca}$ [#] | $[\#]$<br>$\mathcal{Q}_{div}$ |       | <b>t</b> $[s]$ $T_{div}$ $\vert \#$ | $t$ [s] | $T_{diff}$ [#] |
| Scenario <sub>-</sub> $\{0 \rightarrow 1\}$           | 0/0                                                              |              |                               |       |                                     |         |                |
| Scenario <sub>-<math>\{0 \rightarrow 2\}</math></sub> | 12/1                                                             | 45           |                               | 1.77  | 52                                  | 54.99   | 23             |
| Scenario <sub>-</sub> $\{2 \rightarrow 3\}$           | 50/21                                                            | 55           | 21                            | 19.49 | 1269                                | 3423.94 | 1269           |

<span id="page-13-0"></span><sup>3</sup> [https://github.com/VerifAPS/verifaps-lib.](https://github.com/VerifAPS/verifaps-lib)

The first column of Table [3](#page-13-2) denotes the analyzed reconfiguration scenario. The second column contrasts how many change-annotated labels  $\ell_{ca}$  in the reconfigured program exists and how many of those change-annotated labels remain untouched  $\ell_u$  by the test suite of the prior version. This ratio gives an estimate on how suited the previous test suite is to find divergences. The third column denotes the number of test cases  $T_{ca}$  in the previous test suite which exercise any number of change-annotated labels  $\ell_{ca}$  in the change-annotated PLC program. One has to keep in mind that the generated test cases are succinct with regards to the required number of cycles to reach a specific branch (in case of branch coverage). Due to the cyclic nature of the PLC software, test cases which cover deeper nested branches, i.e., branches reachable after a certain amount of cycles, can share a partial prefix with test cases covering already some of the branches on these paths. This is a natural limitation of the SSE approach for cyclic programs resulting in an increased analysis time for phase 1 and phase 2. The fourth column denotes the number of derived divergent contexts and the time it took to complete phase 1 for each representative test case. The fifth column denotes the number of divergent test cases generated from propagating the divergent contextes during BSE using the corresponding triggering test cases as a seed for the concolic execution and the time it took to complete phase 2. The sixth column denotes the number of differencerevealing test cases found by checking the observable behavior of the old and the new version of the program on the divergent test cases.

## **5 Conclusion**

The state of the art for TSA is dominated by DSE techniques [\[3](#page-15-2)]. We implemented a baseline BSE improving scalability issues prevalent in prior work [\[4](#page-15-3)] due to infrequent merging and inefficient storing of the execution contexts. On top of this baseline, we implemented the concept of four-way forking from SSE [\[10\]](#page-16-4) and evaluated the feasibility of this technique on a manually instrumented *regression* benchmark. The number of untouched change-annotated labels in the benchmark of Table [3](#page-13-2) show the limitation of the SSE approach when trying to analyze reconfigurations that introduce new functionality and modify the interfaces of the POUs. As SSE is driven by concrete inputs from an existing test suite, hitting a change is trivially necessary to exercise it. This also means that important divergences can be missed as it strongly depends on the quality of the initial inputs. There has been work that investigated a full exploration of the four-way fork, not only to a predefined bound, but the experiments have shown that it is intractable in general  $[14]$ it does not scale well. Another downside of the SSE approach in the domain of PLC software lies in the search for additional divergent behaviors. Starting a BSE run from the divergence in the new version leads to the coverage of locations that would have been covered with a more succinct prefix. Due to the cyclic nature, the path prefix of the divergence prevents the coverage of the prior branches – however, it is undecidable in general whether this is redundant or not as it would require a procedure to check before the execution, whether that path is difference-revealing or not. To conclude, SSE can be used to generate difference-revealing test cases that are suitable for augmentation of the test suite after a reconfiguration. However, it certainly requires further techniques to reduce the amount of generated differencerevealing test cases to benefit the developer during reconfiguration.

**Outlook.** In future work, we would like to improve our baseline BSE and evaluate more sophisticated merging strategies [\[16](#page-16-12)] or the incorporation of incremental solving [\[12](#page-16-13)]. While merging may prevent an exponential growth of symbolic execution contexts and can boost the efficiency [\[11\]](#page-16-14), the reuse of summaries alleviates the analysis by not doing redundant work for paths through the program we have already seen during execution [\[12](#page-16-13)]. However, summarization and merging are conflicting techniques as checking whether a summary is applicable or not is based on concrete values, a piece of information we would lose through a merge. It remains unclear how to benefit the most from merging and summarization.

**Acknowledgements.** Funded by the Deutsche Forschungsgemeinschaft (DFG, German Research Foundation) under Germany's Excellence Strategy – EXC-2023 Internet of Production – 390621612.

# **References**

- <span id="page-15-1"></span>1. Adiego, B.F., Darvas, D., Viñuela, E.B., Tournier, J.C., Suárez, V.M.G., Blech, J.O.: Modelling and formal verification of timing aspects in large plc programs. IFAC Proc. **47**(3), 3333–3339 (2014). [https://doi.org/10.3182/20140824-6-ZA-](https://doi.org/10.3182/20140824-6-ZA-1003.01279)[1003.01279.](https://doi.org/10.3182/20140824-6-ZA-1003.01279) 19th IFAC World Congress
- <span id="page-15-5"></span>2. Allen, F.E.: Control flow analysis. In: Northcote, R.S. (ed.) Proceedings of a Symposium on Compiler Optimization, Urbana-Champaign, Illinois, USA, 27–28 July 1970, pp. 1–19. ACM (1970). <https://doi.org/10.1145/800028.808479>
- <span id="page-15-2"></span>3. Baldoni, R., Coppa, E., D'Elia, D.C., Demetrescu, C., Finocchi, I.: A survey of symbolic execution techniques. ACM Comput. Surv. **51**(3), 50:1-50:39 (2018). [https://](https://doi.org/10.1145/3182657) [doi.org/10.1145/3182657](https://doi.org/10.1145/3182657)
- <span id="page-15-3"></span>4. Bohlender, D., Simon, H., Friedrich, N., Kowalewski, S., Hauck-Stattelmann, S.: Concolic test generation for PLC programs using coverage metrics. In: Cassandras, C.G., Giua, A., Li, Z. (eds.) 13th International Workshop on Discrete Event Systems, WODES 2016, Xi'an, China, 30 May – 1 June 2016, pp. 432–437. IEEE (2016). <https://doi.org/10.1109/WODES.2016.7497884>
- <span id="page-15-4"></span>5. Godefroid, P.: Compositional dynamic test generation. In: Hofmann, M., Felleisen, M. (eds.) Proceedings of the 34th ACM SIGPLAN-SIGACT Symposium on Principles of Programming Languages, POPL 2007, Nice, France, 17–19 January 2007, pp. 47–54. ACM (2007). <https://doi.org/10.1145/1190216.1190226>
- <span id="page-15-0"></span>6. Grochowski, M., et al.: Formale methoden für rekonfigurierbare cyber-physische systeme in der produktion. at-Automatisierungstechnik **68**(1), 3–14 (2020). [https://](https://doi.org/10.1515/auto-2019-0115) [doi.org/10.1515/auto-2019-0115](https://doi.org/10.1515/auto-2019-0115)
- <span id="page-15-6"></span>7. Gurfinkel, A., Chaki, S.: Boxes: a symbolic abstract domain of boxes. In: Cousot, R., Martel, M. (eds.) SAS 2010. LNCS, vol. 6337, pp. 287–303. Springer, Heidelberg (2010). [https://doi.org/10.1007/978-3-642-15769-1](https://doi.org/10.1007/978-3-642-15769-1_18) 18
- <span id="page-16-5"></span>8. International Electrotechnical Commission: IEC 61131-3:2013 Programmable controllers - Part 3: Programming languages. IEC International Standard IEC 61131- 3:2013 (2013). <https://webstore.iec.ch/publication/4552>
- <span id="page-16-0"></span>9. Jeschke, S., Brecher, C., Song, H., Rawat, D.B. (eds.): Industrial Internet of Things. SSWT, Springer, Cham (2017). <https://doi.org/10.1007/978-3-319-42559-7>
- <span id="page-16-4"></span>10. Kuchta, T., Palikareva, H., Cadar, C.: Shadow symbolic execution for testing software patches. ACM Trans. Softw. Eng. Methodol. **27**(3), 10:1-10:32 (2018). [https://](https://doi.org/10.1145/3208952) [doi.org/10.1145/3208952](https://doi.org/10.1145/3208952)
- <span id="page-16-14"></span>11. Kuznetsov, V., Kinder, J., Bucur, S., Candea, G.: Efficient state merging in symbolic execution. In: Proceedings of the 33rd ACM SIGPLAN Conference on Programming Language Design and Implementation, PLDI 2012, pp. 193–204. Association for Computing Machinery, New York (2012). [https://doi.org/10.1145/2254064.](https://doi.org/10.1145/2254064.2254088) [2254088](https://doi.org/10.1145/2254064.2254088)
- <span id="page-16-13"></span>12. Lin, Y., Miller, T., Søndergaard, H.: Compositional symbolic execution: Incremental solving revisited. In: Potanin, A., Murphy, G.C., Reeves, S., Dietrich, J. (eds.) 23rd Asia-Pacific Software Engineering Conference, APSEC 2016, Hamilton, New Zealand, 6–9 December 2016, pp. 273–280. IEEE Computer Society (2016). [https://](https://doi.org/10.1109/APSEC.2016.046) [doi.org/10.1109/APSEC.2016.046](https://doi.org/10.1109/APSEC.2016.046)
- <span id="page-16-9"></span>13. de Moura, L., Bjørner, N.: Z3: an efficient SMT solver. In: Ramakrishnan, C.R., Rehof, J. (eds.) TACAS 2008. LNCS, vol. 4963, pp. 337–340. Springer, Heidelberg (2008). [https://doi.org/10.1007/978-3-540-78800-3](https://doi.org/10.1007/978-3-540-78800-3_24) 24
- <span id="page-16-11"></span>14. Noller, Y., Nguyen, H.L., Tang, M., Kehrer, T., Grunske, L.: Complete shadow symbolic execution with java pathfinder. ACM SIGSOFT Softw. Eng. Notes **44**(4), 15– 16 (2019). <https://doi.org/10.1145/3364452.33644558>
- <span id="page-16-10"></span>15. PLCopen - Technical Committee 5: Safety software, technical specification, part 1: Concepts and function blocks. Technical report, PLCopen (2020). [https://plcopen.](https://plcopen.org/system/files/downloads/plcopen_safety_part_1_version_2.01.pdf) [org/system/files/downloads/plcopen](https://plcopen.org/system/files/downloads/plcopen_safety_part_1_version_2.01.pdf) safety part 1 version 2.01.pdf
- <span id="page-16-12"></span>16. Sen, K., Necula, G., Gong, L., Choi, W.: MultiSE: multi-path symbolic execution using value summaries. In: Proceedings of the 2015 10th Joint Meeting on Foundations of Software Engineering, ESEC/FSE 2015, pp. 842–853. Association for Computing Machinery, New York (2015). <https://doi.org/10.1145/2786805.2786830>
- <span id="page-16-7"></span>17. Ulewicz, S., Vogel-Heuser, B.: Industrially applicable system regression test prioritization in production automation. IEEE Trans Autom. Sci. Eng. **15**(4), 1839–1851 (2018). <https://doi.org/10.1109/TASE.2018.2810280>
- <span id="page-16-8"></span>18. Weigl, A., Ulbrich, M., Lentzsch, D.: Modular regression verification for reactive systems. In: Margaria, T., Steffen, B. (eds.) ISoLA 2020, Part II. LNCS, vol. 12477, pp. 25–43. Springer, Cham (2020). [https://doi.org/10.1007/978-3-030-61470-6](https://doi.org/10.1007/978-3-030-61470-6_3) 3
- <span id="page-16-1"></span>19. Weyrich, M., Zeller, A.: Testen von industrie-4.0-systemen - wie vernetzte systeme und industrie 4.0 unser verständnis von systemtest und qualitätssicherung ¨andern (2016). [https://www.ias.uni-stuttgart.de/dokumente/vortraege/2016-01-](https://www.ias.uni-stuttgart.de/dokumente/vortraege/2016-01-26_Industrie40_Duesseldorf_v12final.pdf) 26 Industrie40 Duesseldorf [v12final.pdf](https://www.ias.uni-stuttgart.de/dokumente/vortraege/2016-01-26_Industrie40_Duesseldorf_v12final.pdf)
- <span id="page-16-3"></span>20. Weyuker, E.J., Jeng, B.: Analyzing partition testing strategies. IEEE Trans. Softw. Eng. **17**(7), 703–711 (1991). <https://doi.org/10.1109/32.83906>
- <span id="page-16-2"></span>21. Xu, Z., Kim, Y., Kim, M., Cohen, M.B., Rothermel, G.: Directed test suite augmentation: an empirical investigation. Softw. Test. Verif. Reliab. **25**(2), 77–114 (2015). <https://doi.org/10.1002/stvr.1562>
- <span id="page-16-6"></span>22. Yang, G., Person, S., Rungta, N., Khurshid, S.: Directed incremental symbolic execution. ACM Trans. Softw. Eng. Methodol. **24**(1), 3:1-3:42 (2014). [https://doi.org/](https://doi.org/10.1145/2629536) [10.1145/2629536](https://doi.org/10.1145/2629536)
- <span id="page-17-0"></span>23. Yoo, S., Harman, M.: Regression testing minimization, selection and prioritization: a survey. Softw. Test. Verif. Reliab. **22**(2), 67–120 (2012). [https://doi.org/10.1002/](https://doi.org/10.1002/stv.430) [stv.430](https://doi.org/10.1002/stv.430)
- <span id="page-17-1"></span>24. Zeller, A., Jazdi, N., Weyrich, M.: Functional verification of distributed automation systems. Int. J. Adv. Manufact. Technol. **105**(9), 3991–4004 (2019). [https://doi.org/](https://doi.org/10.1007/s00170-019-03791-2) [10.1007/s00170-019-03791-2](https://doi.org/10.1007/s00170-019-03791-2)