The Importance of Software Quality When it Comes to Cars
The Importance of Software Quality When it Comes to Cars
Driverless cars are some of the most attention-grabbing innovations out there. But they could be built on a house of cards. See how technical debt and legacy code are impacting this new venue.
Join the DZone community and get the full member experience.Join For Free
Vehicles have already evolved from mechanical devices into complex integrated technology platforms with embedded software powering all major systems including: engine control, power train, braking, driver assistance, and infotainment. Studies now predict that by 2017, four out of five new cars will have an Internet connection. This “always-on” connectivity will result in new challenges as the line between consumer-grade software for infotainment and safety-critical software gets blurred.
For example, the telematics system provides features such as in-vehicle voice control applications as well as interaction with the GPS system for navigation and traffic features. As we move into the era of increasingly connected and autonomous cars, features such “auto SOS,” which summons help in the event of a crash, will be built on top of this existing telematics architecture.
The news earlier this year of the agreement between major automakers and the U.S. National Highway Traffic Safety Administration to implement automatic emergency braking (AEB) as standard equipment on most vehicles by 2022 signifies another example of this shift from consumer-grade to safety-critical. The AEB systems are controlled by software that powers cameras, radar, proximity sensors and more that all need to operate flawlessly in order to safely stop a vehicle if a driver is slow to respond. This also means that an embedded camera that was previously used for driver assistance (parking for example) will now be part of a safety-critical system.
Figure 1: Software powers all major systems in the modern connected care including engine control, power train, braking, driver assistance, and infotainment.
Insurmountable Quality Issues
Most new software applications are built on legacy code bases. Because a substantial investment of money and time has gone into developing existing applications, there is naturally an interest in leveraging this investment.
The problem with the reuse of existing code is that legacy applications often carry an enormous amount of technical debt (a metaphor for shortcuts taken during the initial design and development of a system). This “debt” is often caused by the continual development of software without the correct quality control processes in place, typically due to the tremendous business pressure to release new versions. The accumulated liability of the technical debt created eventually makes software difficult to maintain.
The key to reducing technical debt and improving quality is to refactor components (the process of restructuring application components without changing its external behavior/API), but developers are often hesitant to do so for fear of breaking existing functionality. One of the biggest impediments to refactoring is the lack of sufficient tests which formalize the existing correct behavior of an application.
Without sufficient tests, it is difficult to refactor an application and not cause regressions in functionality or performance. According to a Gartner study, "a lack of repeatable test cases limits an organization’s ability to demonstrate functional equivalence in an objective, measurable way." This lack of sufficient tests ultimately means that a software application cannot be easily modified to support new features.
Paying Off Technical Debt
Baseline testing, also known as characterization testing, is useful for legacy code bases that have inadequate testing. It is unlikely that developers of an already deployed application will go back and implement all of the low-level tests that should be generated. They rightly feel that the deployed application is “working fine” so why should they spend months of time retesting?
A better option, in this case, is to use automatic test case generation to quickly provide a baseline set of tests that capture and characterize existing application behavior. While these tests do not prove correctness, they do formalize what the application does today, which is very powerful because it allows future changes to be validated to ensure that they do not break existing functionality.
Figure 2: Baseline testing formalizes what an application does today, which allows future changes to be validated to ensure that existing functionality is not broken. Change-based testing can be used to run only the minimum set of test cases needed to show the effect of changes.
Another benefit of having a complete set of baseline tests is that change-based testing (CBT) can be used to reduce total test cycle times. It is not uncommon for complete application testing to take one to two weeks. With change-based testing, small changes can be tested in minutes. Change-based testing computes the minimum set of test cases needed for each code change and runs only those tests.
As a result, developers are able to make incremental changes to the code and ensure that those changes are not breaking the existing behavior of the software. They are also able to do further analysis if something is broken to work out if a bug has been introduced, a capability has been removed that actually should be there, or if there is a bug that should be addressed because it may have other ramifications.
In the Internet of Things-enabled world, a great amount of legacy code bases will find its way onto critical paths in new applications. Without proper software quality methods in place to ensure the integrity of this legacy code, the overall safety of the system could be compromised.
Baseline testing can help reduce technical debt in existing code bases, allow developers to refactor and enhance those code bases with confidence, and ultimately allow the owners of these legacy applications to extract more value.
Opinions expressed by DZone contributors are their own.