Fixed-Point Designer™ User's Guide
Fixed-Point Designer™ User's Guide
User’s Guide
R2014a
How to Contact MathWorks
www.mathworks.com Web
comp.soft-sys.matlab Newsgroup
www.mathworks.com/contact_TS.html Technical Support
508-647-7000 (Phone)
508-647-7001 (Fax)
Trademarks
MATLAB and Simulink are registered trademarks of The MathWorks, Inc. See
www.mathworks.com/trademarks for a list of additional trademarks. Other product or brand
names may be trademarks or registered trademarks of their respective holders.
Patents
MathWorks products are protected by one or more U.S. patents. Please see
www.mathworks.com/patents for more information.
Revision History
March 2013 Online only New for Version 4.0 (R2013a)
September 2013 Online only Revised for Version 4.1 (R2013b)
March 2014 Online only Revised for Version 4.2 (R2014a)
Contents
Scaling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 1-4
iii
Working with fi Objects
2
Ways to Construct fi Objects . . . . . . . . . . . . . . . . . . . . . . . . 2-2
Types of fi Constructors . . . . . . . . . . . . . . . . . . . . . . . . . . . . 2-2
Examples of Constructing fi Objects . . . . . . . . . . . . . . . . . . 2-3
Fixed-Point Topics
3
Set Up Fixed-Point Objects . . . . . . . . . . . . . . . . . . . . . . . . . 3-2
Create Fixed-Point Data . . . . . . . . . . . . . . . . . . . . . . . . . . . . 3-2
iv Contents
Compute Sine and Cosine Using CORDIC Rotation
Kernel . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 3-77
v
fimath Object Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . 4-6
Math, Rounding, and Overflow Properties . . . . . . . . . . . . . 4-6
How Properties are Related . . . . . . . . . . . . . . . . . . . . . . . . . 4-11
Setting fimath Object Properties . . . . . . . . . . . . . . . . . . . . . 4-13
vi Contents
fi Object Display Preferences Using fipref . . . . . . . . . . . 5-5
vii
Working with quantizer Objects
7
Constructing quantizer Objects . . . . . . . . . . . . . . . . . . . . . 7-2
Fixed-Point Conversion
8
Fixed-Point Conversion Workflows . . . . . . . . . . . . . . . . . . 8-2
Manual Workflow . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 8-2
Automated Project Workflow . . . . . . . . . . . . . . . . . . . . . . . . 8-2
viii Contents
Detect Overflows at the Command Line ............. 8-30
ix
Avoid Explicit Double and Single Casts . . . . . . . . . . . . . . . 8-119
x Contents
Set Up C Code Compilation Options . . . . . . . . . . . . . . . . . 9-30
C Code Compiler Configuration Object . . . . . . . . . . . . . . . . 9-30
Compilation Options Modification at the Command Line
Using Dot Notation . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9-30
How fiaccel Resolves Conflicting Options . . . . . . . . . . . . . . 9-31
xi
Types of Run-Time Checks . . . . . . . . . . . . . . . . . . . . . . . . . . 9-62
When to Disable Run-Time Checks . . . . . . . . . . . . . . . . . . . 9-63
How to Disable Run-Time Checks . . . . . . . . . . . . . . . . . . . . 9-63
xii Contents
Code Generation Readiness Tool . . . . . . . . . . . . . . . . . . . . 9-114
What Information Does the Code Generation Readiness
Tool Provide? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9-114
Summary Tab . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9-115
Code Structure Tab . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9-117
See Also . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9-120
xiii
Calling Functions for Code Generation
11
Resolution of Function Calls for Code Generation . . . . 11-2
Key Points About Resolving Function Calls . . . . . . . . . . . . 11-4
Compile Path Search Order . . . . . . . . . . . . . . . . . . . . . . . . . 11-4
When to Use the Code Generation Path . . . . . . . . . . . . . . . 11-5
xiv Contents
Generate Code for MATLAB Value Classes . . . . . . . . . . . 12-9
xv
Apply Operations to a Variable Number of
Arguments . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 14-4
When to Force Loop Unrolling . . . . . . . . . . . . . . . . . . . . . . . 14-4
Using Variable Numbers of Arguments in a for-Loop . . . . 14-5
xvi Contents
Reuse the Same Variable with Different Properties . . . 15-11
When You Can Reuse the Same Variable with Different
Properties . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15-11
When You Cannot Reuse Variables . . . . . . . . . . . . . . . . . . . 15-12
Limitations of Variable Reuse . . . . . . . . . . . . . . . . . . . . . . . 15-14
xvii
Variable-Size Support for Code Generation . . . . . . . . . . . . 16-12
xviii Contents
switch Statement with Enumerated Data Types . . . . . . . . 17-15
while Statement with Enumerated Data Types . . . . . . . . . 17-18
xix
Modularize MATLAB Code ......................... 19-6
xx Contents
Prevent Code Generation for Unused Execution
Paths . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 19-33
Prevent Code Generation When Local Variable Controls
Flow . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 19-33
Prevent Code Generation When Input Variable Controls
Flow . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 19-34
xxi
Restriction When Using struct . . . . . . . . . . . . . . . . . . . . . . . 20-4
Restrictions When Defining Scalar Structures by
Assignment . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 20-4
Adding Fields in Consistent Order on Each Control Flow
Path . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 20-4
Restriction on Adding New Fields After First Use . . . . . . . 20-5
xxii Contents
Control Flow in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-157
Data and File Management in MATLAB . . . . . . . . . . . . . . . 21-158
Data Types in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-161
Desktop Environment in MATLAB . . . . . . . . . . . . . . . . . . . 21-162
Discrete Math in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . . 21-162
DSP System Toolbox . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-163
Error Handling in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . 21-171
Exponents in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-171
Filtering and Convolution in MATLAB . . . . . . . . . . . . . . . . 21-172
Fixed-Point Designer . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-173
Histograms in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-183
Image Acquisition Toolbox . . . . . . . . . . . . . . . . . . . . . . . . . . 21-183
Image Processing Toolbox . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-183
Input and Output Arguments in MATLAB . . . . . . . . . . . . . 21-189
Interpolation and Computational Geometry in MATLAB . . 21-189
Linear Algebra in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . 21-193
Logical and Bit-Wise Operations in MATLAB . . . . . . . . . . 21-193
MATLAB Compiler . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-194
Matrices and Arrays in MATLAB . . . . . . . . . . . . . . . . . . . . 21-194
Nonlinear Numerical Methods in MATLAB . . . . . . . . . . . . 21-202
Numerical Integration and Differentiation in MATLAB . . 21-202
Optimization Functions in MATLAB . . . . . . . . . . . . . . . . . . 21-203
Phased Array System Toolbox . . . . . . . . . . . . . . . . . . . . . . . 21-204
Polynomials in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-212
Programming Utilities in MATLAB . . . . . . . . . . . . . . . . . . . 21-212
Relational Operators in MATLAB . . . . . . . . . . . . . . . . . . . . 21-213
Rounding and Remainder Functions in MATLAB . . . . . . . 21-213
Set Operations in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . 21-214
Signal Processing in MATLAB . . . . . . . . . . . . . . . . . . . . . . . 21-218
Signal Processing Toolbox . . . . . . . . . . . . . . . . . . . . . . . . . . 21-219
Special Values in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . 21-224
Specialized Math in MATLAB . . . . . . . . . . . . . . . . . . . . . . . 21-224
Statistics in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-225
Statistics Toolbox . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-226
String Functions in MATLAB . . . . . . . . . . . . . . . . . . . . . . . 21-235
Structures in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 21-237
Trigonometry in MATLAB . . . . . . . . . . . . . . . . . . . . . . . . . . 21-237
xxiii
Code Generation for Variable-Size Data
22
What Is Variable-Size Data? . . . . . . . . . . . . . . . . . . . . . . . . 22-2
xxiv Contents
Incompatibilities with MATLAB in Variable-Size
Support for Code Generation . . . . . . . . . . . . . . . . . . . . . 22-27
Incompatibility with MATLAB for Scalar Expansion . . . . . 22-27
Incompatibility with MATLAB in Determining Size of
Variable-Size N-D Arrays . . . . . . . . . . . . . . . . . . . . . . . . . 22-29
Incompatibility with MATLAB in Determining Size of
Empty Arrays . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 22-30
Incompatibility with MATLAB in Determining Class of
Empty Arrays . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 22-31
Incompatibility with MATLAB in Vector-Vector
Indexing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 22-32
Incompatibility with MATLAB in Matrix Indexing
Operations for Code Generation . . . . . . . . . . . . . . . . . . . 22-33
Incompatibility with MATLAB in Concatenating
Variable-Size Matrices . . . . . . . . . . . . . . . . . . . . . . . . . . . 22-34
Dynamic Memory Allocation Not Supported for MATLAB
Function Blocks . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 22-35
Primary Functions
23
Primary Function Input Specification . . . . . . . . . . . . . . . 23-2
Why You Must Specify Input Properties . . . . . . . . . . . . . . . 23-2
Properties to Specify . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 23-2
Rules for Specifying Properties of Primary Inputs . . . . . . . 23-4
Methods for Defining Properties of Primary Inputs . . . . . . 23-5
Define Input Properties by Example at the Command
Line . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 23-5
Specify Constant Inputs at the Command Line . . . . . . . . . 23-8
Specify Variable-Size Inputs at the Command Line . . . . . . 23-10
xxv
Specifying General Properties of Primary Inputs . . . . . . . . 23-20
Specifying Properties of Primary Fixed-Point Inputs . . . . . 23-20
Specifying Class and Size of Scalar Structure . . . . . . . . . . . 23-21
Specifying Class and Size of Structure Array . . . . . . . . . . . 23-22
System Objects
25
Create System Objects . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 25-2
Create a System object . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 25-2
Change a System object Property . . . . . . . . . . . . . . . . . . . . 25-3
Check if a System object Property Has Changed . . . . . . . . 25-3
Run a System object . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 25-3
Display Available System Objects . . . . . . . . . . . . . . . . . . . . 25-3
xxvi Contents
Find Help and Examples for System Objects . . . . . . . . . 25-12
xxvii
Specify the Output Data Type and Scaling . . . . . . . . . . . . . 26-22
Specify Fixed-Point Data Types with the Data Type
Assistant . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 26-24
Rounding . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 26-27
Overflow Handling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 26-28
Lock the Output Data Type Setting . . . . . . . . . . . . . . . . . . . 26-28
Real-World Values Versus Stored Integer Values . . . . . . . 26-28
xxviii Contents
Constant Scaling for Best Precision . . . . . . . . . . . . . . . . . . . 27-15
Fixed-Point Data Type and Scaling Notation . . . . . . . . . . . 27-18
Scaled Doubles . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 27-20
Use Scaled Doubles to Avoid Precision Loss . . . . . . . . . . . . 27-22
Display Port Data Types . . . . . . . . . . . . . . . . . . . . . . . . . . . . 27-25
Arithmetic Operations
28
Fixed-Point Arithmetic Operations . . . . . . . . . . . . . . . . . . 28-2
Precision . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-3
Limitations on Precision . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-3
Rounding . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-4
Choose a Rounding Mode . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-4
Rounding Modes for Fixed-Point Simulink Blocks . . . . . . . 28-5
Rounding Mode: Ceiling . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-7
Rounding Mode: Convergent . . . . . . . . . . . . . . . . . . . . . . . . 28-8
Rounding Mode: Floor . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-10
Rounding Mode: Nearest . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-11
Rounding Mode: Round . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-12
Rounding Mode: Simplest . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-14
Rounding Mode: Zero . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-17
Pad with Trailing Zeros . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-19
Limitations on Precision and Errors . . . . . . . . . . . . . . . . . . 28-20
Maximize Precision . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-21
Net Slope and Net Bias Precision . . . . . . . . . . . . . . . . . . . . 28-21
Detect Net Slope and Bias Precision Issues . . . . . . . . . . . . 28-24
Detect Fixed-Point Constant Precision Loss . . . . . . . . . . . . 28-25
Range . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-27
Limitations on Range . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-27
xxix
What Are Saturation and Wrapping? . . . . . . . . . . . . . . . . . 28-28
Saturation and Wrapping . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-28
Guard Bits . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 28-31
Determine the Range of Fixed-Point Numbers . . . . . . . . . . 28-31
Realization Structures
29
Realizing Fixed-Point Digital Filters . . . . . . . . . . . . . . . . 29-2
Introduction . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 29-2
Realizations and Data Types . . . . . . . . . . . . . . . . . . . . . . . . 29-2
xxx Contents
Targeting an Embedded Processor . . . . . . . . . . . . . . . . . . 29-4
Introduction . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 29-4
Size Assumptions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 29-4
Operation Assumptions . . . . . . . . . . . . . . . . . . . . . . . . . . . . 29-4
Design Rules . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 29-5
Fixed-Point Advisor
30
Preparation for Fixed-Point Conversion . . . . . . . . . . . . . 30-2
Introduction . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30-2
Best Practices . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30-2
Run the Fixed-Point Advisor . . . . . . . . . . . . . . . . . . . . . . . . 30-4
Fix a Task Failure . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30-5
Manually Fixing Failures . . . . . . . . . . . . . . . . . . . . . . . . . . . 30-6
Automatically Fixing Failures . . . . . . . . . . . . . . . . . . . . . . . 30-6
Batch Fixing Failures . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30-7
Restore Points . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30-7
Save a Restore Point . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30-8
Load a Restore Point . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30-9
xxxi
Fixed-Point Tool
31
Fixed-Point Tool . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 31-2
Introduction to the Fixed-Point Tool . . . . . . . . . . . . . . . . . . 31-2
Using the Fixed-Point Tool . . . . . . . . . . . . . . . . . . . . . . . . . . 31-2
xxxii Contents
View Signal Names in Fixed-Point Tool . . . . . . . . . . . . . . 31-39
xxxiii
Producing Lookup Table Data
33
Producing Lookup Table Data . . . . . . . . . . . . . . . . . . . . . . 33-2
xxxiv Contents
Before Proposing Data Types for Your Model . . . . . . . . 34-3
xxxv
Propose Fraction Lengths . . . . . . . . . . . . . . . . . . . . . . . . . . 34-43
Propose Fraction Lengths . . . . . . . . . . . . . . . . . . . . . . . . . . . 34-43
About the Feedback Controller Example Model . . . . . . . . . 34-44
Propose Fraction Lengths Using Simulation Range
Data . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 34-50
Range Analysis
35
How Range Analysis Works . . . . . . . . . . . . . . . . . . . . . . . . . 35-2
Analyzing a Model with Range Analysis . . . . . . . . . . . . . . . 35-2
Automatic Stubbing . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 35-5
Model Compatibility with Range Analysis . . . . . . . . . . . . . 35-6
xxxvi Contents
Derive Ranges at the Subsystem Level . . . . . . . . . . . . . . . 35-10
Deriving Ranges at the Subsystem Level . . . . . . . . . . . . . . 35-10
Derive Ranges at the Subsystem Level . . . . . . . . . . . . . . . . 35-11
xxxvii
Working with Bus Objects in the Fixed-Point
Workflow
36
Refine Data Types of a Model with Buses Using
Simulation Data . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 36-2
Open and Simulate the Model . . . . . . . . . . . . . . . . . . . . . . . 36-2
Use Data Type Override to Resolve Overflows . . . . . . . . . . 36-3
Propose New Fraction Lengths . . . . . . . . . . . . . . . . . . . . . . 36-6
Examine and Apply Proposed Types . . . . . . . . . . . . . . . . . . 36-6
Verify New Fixed-Point Settings . . . . . . . . . . . . . . . . . . . . . 36-9
Save the Model and New Bus Data Types . . . . . . . . . . . . . . 36-12
Code Generation
37
Generating and Deploying Production Code . . . . . . . . . 37-2
xxxviii Contents
Languages . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37-3
Data Types . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37-3
Rounding Modes . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37-3
Overflow Handling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37-3
Blocks . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37-4
Scaling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37-4
xxxix
Use the Model Advisor to Optimize Fixed-Point
Operations in Generated Code . . . . . . . . . . . . . . . . . . . . 37-41
xl Contents
Troubleshooting
39
Frequently Asked Questions About Fixed-Point
Numbers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 39-2
What Is the Difference Between Fixed-Point and Built-In
Integer Types? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 39-2
Negative Fraction Length . . . . . . . . . . . . . . . . . . . . . . . . . . . 39-2
Fraction Length Greater Than Word Length . . . . . . . . . . . 39-4
fi Constructor Does Not Follow globalfimath Rules . . . . . . 39-5
xli
What to Do When a Block Is Not Supported For
Fixed-Point Conversion . . . . . . . . . . . . . . . . . . . . . . . . . . 39-18
Which Simulink Blocks Are Supported by the Fixed-Point
Tool . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 39-18
What to Do When a Block Is Not Supported for Fixed-Point
Conversion . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 39-18
xlii Contents
Set the Output Port Data Type . . . . . . . . . . . . . . . . . . . . . . A-27
Interpret an Input Value . . . . . . . . . . . . . . . . . . . . . . . . . . . A-28
Write an Output Value . . . . . . . . . . . . . . . . . . . . . . . . . . . . . A-30
Determine Output Type Using the Input Type . . . . . . . . . . A-31
xliii
xliv Contents
Fixed-Point Designer for
MATLAB Code
1
Fixed-Point Concepts
A fixed-point data type is characterized by the word length in bits, the position
of the binary point, and whether it is signed or unsigned. The position of
the binary point is the means by which fixed-point values are scaled and
interpreted.
bwl 1 bwl 2 b5 b4 b3 b2 b1 b0
where
1-2
Fixed-Point Data Types
• Sign/magnitude
• One’s complement
• Two’s complement
1-3
1 Fixed-Point Concepts
Scaling
Fixed-point numbers can be encoded according to the scheme
The integer is sometimes called the stored integer. This is the raw binary
number, in which the binary point assumed to be at the far right of the word.
In Fixed-Point Designer documentation, the negative of the fixed exponent is
often referred to as the fraction length.
The slope and bias together represent the scaling of the fixed-point number.
In a number with zero bias, only the slope affects the scaling. A fixed-point
number that is only scaled by binary point position is equivalent to a number
in [Slope Bias] representation that has a bias equal to zero and a slope
adjustment factor equal to one. This is referred to as binary point-only scaling
or power-of-two scaling:
or
1-4
Scaling
Note For examples of binary point-only scaling, see the Fixed-Point Designer
Binary-Point Scaling example.
For an example of how to compute slope and bias in MATLAB®, see “Slope
Bias Scaling” on page 1-6
1-5
1 Fixed-Point Concepts
lower_bound = 999;
upper_bound = 1000;
is_signed = true;
word_length = 16;
To find the range of a fi object with a specified word length and signedness,
use the range function.
1-6
Slope Bias Scaling
To find the slope, or precision, call the first element of the slope-bias vector, x.
slope = x(1)
slope =
1.52590218966964e-05
bias = x(2)
bias =
999.500007629511
T =
1-7
1 Fixed-Point Concepts
a = fi(999.255, T)
a =
999.254993514916
Verify that the fi object that you created has the correct specifications by
finding the range of a.
range(a)
ans =
999 1000
1-8
Precision and Range
Note You must pay attention to the precision and range of the fixed-point
data types and scalings you choose in order to know whether rounding
methods will be invoked or if overflows or underflows will occur.
Range
The range is the span of numbers that a fixed-point data type and scaling
can represent. The range of representable numbers for a two’s complement
fixed-point number of word length wl , scaling S and bias B is illustrated
below:
S. ( 2wl 1
)+ B B (
S . 2wl 1
1 +B)
negative numbers positive numbers
For both signed and unsigned fixed-point numbers of any data type, the
number of different bit patterns is 2wl.
2wl 1
0 2wl 1
1
1-9
1 Fixed-Point Concepts
Overflow Handling
Because a fixed-point data type represents numbers within a finite range,
overflows and underflows can occur if the result of an operation is larger or
smaller than the numbers in that range.
When you create a fi object, any overflows are saturated. The OverflowMode
property of the default fimath is saturate. You can log overflows and
underflows by setting the LoggingMode property of the fipref object to on.
Refer to “LoggingMode” for more information.
Precision
The precision of a fixed-point number is the difference between successive
values representable by its data type and scaling, which is equal to the value
of its least significant bit. The value of the least significant bit, and therefore
the precision of the number, is determined by the number of fractional bits.
A fixed-point value can be represented to within half of the precision of its
data type and scaling.
For example, a fixed-point representation with four bits to the right of the
binary point has a precision of 2-4 or 0.0625, which is the value of its least
significant bit. Any number within the range of this data type and scaling can
be represented to within (2-4)/2 or 0.03125, which is half the precision. This is
an example of representing a number with finite precision.
Rounding Methods
When you represent numbers with finite precision, not every number in the
available range can be represented exactly. If a number cannot be represented
exactly by the specified data type and scaling, a rounding method is used to
cast the value to a representable number. Although precision is always lost
in the rounding operation, the cost of the operation and the amount of bias
that is introduced depends on the rounding method itself. To provide you with
1-10
Precision and Range
greater flexibility in the trade-off between cost and bias, Fixed-Point Designer
software currently supports the following rounding methods:
1-11
1 Fixed-Point Concepts
Note The cost estimates provided here are hardware independent. Some
processors have rounding modes built-in, so consider carefully the hardware
you are using before calculating the true cost of each rounding mode.
• Bias — What is the expected value of the rounded values minus the original
( )
values: Ε ˆ − ?
- ( )
Ε ˆ − < 0 — The rounding method introduces a negative bias.
- ( )
Ε ˆ − = 0 — The rounding method is unbiased.
- ( )
Ε ˆ − > 0 — The rounding method introduces a positive bias.
• Possibility of Overflow — Does the rounding method introduce the
possibility of overflow?
- Yes — The rounded values may exceed the minimum or maximum
representable value.
- No — The rounded values will never exceed the minimum or maximum
representable value.
1-12
Precision and Range
1-13
1 Fixed-Point Concepts
Arithmetic Operations
In this section...
“Modulo Arithmetic” on page 1-14
“Two’s Complement” on page 1-15
“Addition and Subtraction” on page 1-16
“Multiplication” on page 1-17
“Casts” on page 1-23
Note These sections will help you understand what data type and scaling
choices result in overflows or a loss of precision.
Modulo Arithmetic
Binary math is based on modulo arithmetic. Modulo arithmetic uses only
a finite set of numbers, wrapping the results of any calculations that fall
outside the given set back into the set.
For example, the common everyday clock uses modulo 12 arithmetic. Numbers
in this system can only be 1 through 12. Therefore, in the “clock” system, 9
plus 9 equals 6. This can be more easily visualized as a number circle:
1-14
Arithmetic Operations
12 12
11 1 11 1
10 2 10 2
9 3 9 3
8 4 8 4
7 5 7 5
6 6
... equals 6.
Similarly, binary math can only use the numbers 0 and 1, and any arithmetic
results that fall outside this range are wrapped “around the circle” to either 0
or 1.
Two’s Complement
Two’s complement is a way to interpret a binary number. In two’s
complement, positive numbers always start with a 0 and negative numbers
always start with a 1. If the leading bit of a two’s complement number is 0,
the value is obtained by calculating the standard binary value of the number.
If the leading bit of a two’s complement number is 1, the value is obtained by
assuming that the leftmost bit is negative, and then calculating the binary
value of the number. For example,
01 = (0 + 20 ) = 1
11 = (( −2 ) + ( 2 )) = (−2 + 1) = −1
1 0
1-15
1 Fixed-Point Concepts
For example, consider taking the negative of 11010 (-6). First, take the one’s
complement of the number, or flip the bits:
11010 → 00101
00101
+1
00110 (6)
For example, consider the addition of 010010.1 (18.5) with 0110.110 (6.75):
010010.1 (18.5)
+0110.110 (6.75)
011001.010 (25.25)
Carry bit
is discarded
1-16
Arithmetic Operations
The default fimath has a value of 1 (true) for the CastBeforeSum property.
This casts addends to the sum data type before addition. Therefore, no further
shifting is necessary during the addition to line up the binary points.
If CastBeforeSum has a value of 0 (false), the addends are added with full
precision maintained. After the addition the sum is then quantized.
Multiplication
The multiplication of two’s complement fixed-point numbers is directly
analogous to regular decimal multiplication, with the exception that the
intermediate results must be sign-extended so that their left sides align
before you add them together.
For example, consider the multiplication of 10.11 (-1.25) with 011 (3):
1-17
1 Fixed-Point Concepts
Real-Real Multiplication. The following diagram shows the data types used
by the toolbox in the multiplication of two real numbers. The software returns
the output of this operation in the product data type, which is governed by the
fimath object ProductMode property.
1-18
Arithmetic Operations
When the fimath object CastBeforeSum property is true, the casts to the
sum data type are present after the multipliers in the preceding diagram. In
C code, this is equivalent to
acc=ac;
acc-=bd;
acc=ad;
acc+=bc;
for the adder, where acc is the accumulator. When the CastBeforeSum
property is false, the casts are not present, and the data remains in the
product data type before the subtraction and addition operations.
1-19
1 Fixed-Point Concepts
F = fimath('ProductMode','FullPrecision',...
'SumMode','FullPrecision')
T1 = numerictype('WordLength',24,'FractionLength',20)
T2 = numerictype('WordLength',16,'FractionLength',10)
Real*Real. Notice that the word length and fraction length of the result z
are equal to the sum of the word lengths and fraction lengths, respectively,
of the multiplicands. This is because the fimath SumMode and ProductMode
properties are set to FullPrecision:
P = fipref;
P.FimathDisplay = 'none';
x = fi(5, T1, F)
x =
y = fi(10, T2, F)
y =
10
z = x*y
1-20
Arithmetic Operations
z =
50
Real*Complex. Notice that the word length and fraction length of the result
z are equal to the sum of the word lengths and fraction lengths, respectively,
of the multiplicands. This is because the fimath SumMode and ProductMode
properties are set to FullPrecision:
x = fi(5,T1,F)
x =
y = fi(10+2i,T2,F)
y =
10.0000 + 2.0000i
1-21
1 Fixed-Point Concepts
z = x*y
z =
50.0000 +10.0000i
x = fi(5+6i,T1,F)
x =
5.0000 + 6.0000i
y = fi(10+2i,T2,F)
y =
10.0000 + 2.0000i
1-22
Arithmetic Operations
z = x*y
z =
38.0000 +70.0000i
Casts
The fimath object allows you to specify the data type and scaling of
intermediate sums and products with the SumMode and ProductMode
properties. It is important to keep in mind the ramifications of each cast
when you set the SumMode and ProductMode properties. Depending upon the
data types you select, overflow and/or rounding might occur. The following
two examples demonstrate cases where overflow and rounding can occur.
Note For more examples of casting, see “Cast fi Objects” on page 2-13.
1-23
1 Fixed-Point Concepts
destination
As the diagram shows, the source bits are shifted up so that the binary point
matches the destination binary point position. The highest source bit does
not fit, so overflow might occur and the result can saturate or wrap. The
empty bits at the low end of the destination data type are padded with either
0’s or 1’s:
• If overflow does not occur, the empty bits are padded with 0’s.
• If wrapping occurs, the empty bits are padded with 0’s.
• If saturation occurs,
- The empty bits of a positive number are padded with 1’s.
- The empty bits of a negative number are padded with 0’s.
You can see that even with a cast from a shorter data type to a longer data
type, overflow can still occur. This can happen when the integer length of
the source data type (in this case two) is longer than the integer length of
the destination data type (in this case one). Similarly, rounding might be
necessary even when casting from a shorter data type to a longer data type, if
the destination data type and scaling has fewer fractional bits than the source.
1-24
Arithmetic Operations
destination
As the diagram shows, the source bits are shifted down so that the binary
point matches the destination binary point position. There is no value for
the highest bit from the source, so sign extension is used to fill the integer
portion of the destination data type. The bottom five bits of the source do not
fit into the fraction length of the destination. Therefore, precision can be
lost as the result is rounded.
In this case, even though the cast is from a longer data type to a shorter
data type, all the integer bits are maintained. Conversely, full precision can
be maintained even if you cast to a shorter data type, as long as the fraction
length of the destination data type is the same length or longer than the
fraction length of the source data type. In that case, however, bits are lost
from the high end of the result and overflow can occur.
The worst case occurs when both the integer length and the fraction length of
the destination data type are shorter than those of the source data type and
scaling. In that case, both overflow and a loss of precision can occur.
1-25
1 Fixed-Point Concepts
Note The sections in this topic compare the fi object with fixed-point data
types and operations in C. In these sections, the information on ANSI® C
is adapted from Samuel P. Harbison and Guy L. Steele Jr., C: A Reference
Manual, 3rd ed., Prentice Hall, 1991.
1-26
fi Objects and C Integer Data Types
1-27
1 Fixed-Point Concepts
Unary Conversions
Unary conversions dictate whether and how a single operand is converted
before an operation is performed. This section discusses unary conversions
in ANSI C and of fi objects.
1
If type int cannot represent all the values of the original data type without
overflow, the converted type is unsigned int.
1-28
fi Objects and C Integer Data Types
Binary Conversions
This section describes the conversions that occur when the operands of a
binary operator are different data types.
1-29
1 Fixed-Point Concepts
Type of Other
Type of One Operand Operand ANSI C Conversion
long double Any long double
double Any double
float Any float
unsigned long Any unsigned long
long unsigned long or unsigned
long1
long int long
unsigned int or unsigned unsigned
int int int
1
Type long is only used if it can represent all values of type unsigned.
1-30
fi Objects and C Integer Data Types
1-31
1 Fixed-Point Concepts
Overflow Handling
The following sections compare how ANSI C and Fixed-Point Designer
software handle overflows.
fi Overflow Handling
Addition and multiplication with fi objects yield results that can be exactly
represented by a fi object, up to word lengths of 65,535 bits or the available
memory on your machine. This is not true of division, however, because many
ratios result in infinite binary expressions. You can perform division with fi
objects using the divide function, which requires you to explicitly specify the
numeric type of the result.
The conditions under which a fi object overflows and the results then
produced are determined by the associated fimath object. You can specify
certain overflow characteristics separately for sums (including differences)
and products. Refer to the following table:
1-32
fi Objects and C Integer Data Types
1-33
1 Fixed-Point Concepts
1-34
fi Objects and C Integer Data Types
1-35
1 Fixed-Point Concepts
1-36
2
Types of fi Constructors
You can create fi objects using Fixed-Point Designer software in any of the
following ways:
To get started, to create a fi object with the default data type and a value of 0:
a = fi(0)
a =
To see all of the fi, sfi, and ufi constructor syntaxes, refer to the respective
reference pages.
2-2
Ways to Construct fi Objects
• fi
• sfi
• ufi
a =
3.1415
RoundingMethod: Floor
OverflowAction: Wrap
2-3
2 Working with fi Objects
ProductMode: FullPrecision
SumMode: FullPrecision
You do not have to specify every fimath object property in the fi constructor.
The fi object uses default values for all unspecified fimath object properties.
• If you specify at least one fimath object property in the fi constructor, the
fi object has a local fimath object. The fi object uses default values for
the remaining unspecified fimath object properties.
• If you do not specify any fimath object properties in the fi object
constructor, the fi object uses default fimath values.
T = numerictype
T =
a = fi(pi, T)
a =
1.0000
You can also use a fimath object with a numerictype object to define a fi
object:
2-4
Ways to Construct fi Objects
F = fimath('RoundingMethod', 'Nearest',...
'OverflowAction', 'Saturate',...
'ProductMode','FullPrecision',...
'SumMode','FullPrecision')
F =
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: FullPrecision
a = fi(pi, T, F)
a =
1.0000
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: FullPrecision
2-5
2 Working with fi Objects
any numerictype object properties, the word length of the fi object defaults
to 16 bits. The fraction length is determined by best precision scaling:
F = fimath('RoundingMethod', 'Nearest',...
'OverflowAction', 'Saturate',...
'ProductMode','FullPrecision',...
'SumMode','FullPrecision')
F =
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: FullPrecision
F.OverflowAction = 'Wrap'
F =
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
a = fi(pi, F)
a =
3.1416
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
2-6
Ways to Construct fi Objects
You can also create fi objects using a fimath object while specifying various
numerictype properties at creation time:
b = fi(pi, 0, F)
b =
3.1416
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
c = fi(pi, 0, 8, F)
c =
3.1406
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
d = fi(pi, 0, 8, 6, F)
d =
3.1406
2-7
2 Working with fi Objects
RoundingMethod: Nearest
OverflowAction: wrap
ProductMode: FullPrecision
SumMode: FullPrecision
For example, to create a signed fi object with a value of pi, a word length of
16 bits and a fraction length of 13 bits:
1 On the Home tab, in the File section, click New > Script to open the
MATLAB Editor.
2 On the Editor tab, in the Edit section, click in the Insert button
group. Click Insert fi... to open the Insert fi Constructor dialog box.
3 Use the edit boxes and drop-down menus to specify the following properties
of the fi object:
• Value = pi
• Data type mode = Fixed-point: binary point scaling
• Signedness = Signed
• Word length = 16
• Fraction length = 13
2-8
Ways to Construct fi Objects
4 To insert the fi object constructor string in your file, place your cursor at
the desired location in the file, and click OK on the Insert fi Constructor
dialog box. Clicking OK closes the Insert fi Constructor dialog box and
automatically populates the fi object constructor string in your file:
T =
2-9
2 Working with fi Objects
Now, create the following fi object in which you specify the numerictype
property after the Signed property, so that the resulting fi object is signed:
a = fi(pi,'Signed',false,'numerictype',T)
a =
1.9999
Contrast the fi object in this code sample with the fi object in the following
code sample. The numerictype property in the following code sample is
specified before the Signed property, so the resulting fi object is unsigned:
b = fi(pi,'numerictype',T,'Signed',false)
b =
3.1416
Copying a fi Object
To copy a fi object, simply use assignment:
a = fi(pi)
a =
3.1416
2-10
Ways to Construct fi Objects
Signedness: Signed
WordLength: 16
FractionLength: 13
b = a
b =
3.1416
function T = mytypes(dt)
switch dt
case 'double'
T.b = double([]);
T.x = double([]);
T.y = double([]);
case 'fixed16'
T.b = fi([],true,16,15);
T.x = fi([],true,16,15);
T.y = fi([],true,16,14);
end
end
Cast the variables in the algorithm to the data types in the types table as
described in “Manual Fixed-Point Conversion Best Practices” on page 8-89.
function [y,z]=myfilter(b,x,z,T)
y = zeros(size(x),'like',T.y);
for n=1:length(x)
z(:) = [x(n); z(1:end-1)];
2-11
2 Working with fi Objects
y(n) = b * z;
end
end
In a separate test file, set up input data to feed into your algorithm, and
specify the data types of the inputs.
% Test inputs
b = fir1(11,0.25);
t = linspace(0,10*pi,256)';
x = sin((pi/16)*t.^2);
% Linear chirp
% Cast inputs
T=mytypes('fixed16');
b=cast(b,'like',T.b);
x=cast(x,'like',T.x);
z=zeros(size(b'),'like',T.x);
% Run
[y,z] = myfilter(b,x,z,T);
2-12
Cast fi Objects
Cast fi Objects
In this section...
“Overwriting by Assignment” on page 2-13
“Ways to Cast with MATLAB Software” on page 2-13
Overwriting by Assignment
Because MATLAB software does not have type declarations, an assignment
like A = B replaces the type and content of A with the type and content of B. If
A does not exist at the time of the assignment, MATLAB creates the variable
A and assigns it the same type and value as B. Such assignment happens
with all types in MATLAB—objects and built-in types alike—including fi,
double, single, int8, uint8, int16, etc.
For example, the following code overwrites the value and int8 type of A with
the value and int16 type of B:
A = int8(0);
B = int16(32767);
A = B
A =
32767
class(A)
ans =
int16
2-13
2 Working with fi Objects
A = int8(0);
B = int16(32767);
A(:) = B
A =
127
class(A)
ans =
int8
fipref('NumericTypeDisplay', 'short');
A = fi(0, true, 8, 0);
B = fi(32767, true, 16, 0);
A(:) = B
A =
127
s8,0
2-14
Cast fi Objects
B = int16(32767);
A = int8(B)
A =
127
class(A)
ans =
int8
B =
32767
s16,0
A =
127
s8,0
2-15
2 Working with fi Objects
T8 = numerictype(1,8,0)
T8 =
T16 = numerictype(1,16,0)
T16 =
B = fi(32767,T16)
B =
32767
s16,0
A = fi(B, T8)
A =
127
s8,0
2-16
Cast fi Objects
B =
A =
To verify that the underlying data has not changed, compare the binary
representations of A and B:
binary_B = bin(B)
binary_A = bin(A)
binary_A =
2-17
2 Working with fi Objects
binary_B =
In the following example, a is cast to the data type of b. The output, c, has the
same numerictype and fimath properties as b, and the value of a.
a = pi;
b = fi([],1,16,13,'RoundingMethod',Floor);
c= cast(a,'like',b)
c =
3.1415
RoundingMethod: Floor
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: FullPrecision
Using this syntax allows you to specify data types separately from your
algorithmic code as described in “Manual Fixed-Point Conversion Best
Practices” on page 8-89.
2-18
fi Object Properties
fi Object Properties
In this section...
“Data Properties” on page 2-19
“fimath Properties” on page 2-19
“numerictype Properties” on page 2-21
“Setting fi Object Properties” on page 2-22
Data Properties
The data properties of a fi object are always writable.
To learn more about these properties, see “fi Object Properties” in the
Fixed-Point Designer Reference.
fimath Properties
In general, the fimath properties associated with fi objects depend on how
you create the fi object:
2-19
2 Working with fi Objects
• CastBeforeSum — Whether both operands are cast to the sum data type
before addition
2-20
fi Object Properties
numerictype Properties
When you create a fi object, a numerictype object is also automatically
created as a property of the fi object:
2-21
2 Working with fi Objects
There are two ways to specify properties for fi objects in Fixed-Point Designer
software. Refer to the following sections:
a =
3.1416
2-22
fi Object Properties
RoundingMethod: Convergent
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
To set the stored integer value of a fi object, use the parameter/value pair
for the 'int' property when you create the object. For example, create a
signed fi object with a stored integer value of 4, 16-bit word length, and
15-bit fraction length.
x = fi(0,true,16,15,'int',4);
x.int
ans =
a.WordLength
ans =
16
2-23
2 Working with fi Objects
a.OverflowAction = 'Wrap'
a =
3.1416
RoundingMethod: Convergent
OverflowAction: wrap
ProductMode: FullPrecision
SumMode: FullPrecision
If you have a fi object b with a local fimath object, you can remove the local
fimath object and force b to use default fimath values:
b =
3.1415
RoundingMethod: Floor
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: FullPrecision
b.fimath = []
b =
3.1415
2-24
fi Object Properties
isfimathlocal(b)
ans =
0
2-25
2 Working with fi Objects
fi Object Functions
In addition to functions that operate on fi objects, you can use the following
functions to access data in a fi object using dot notation.
• bin
• data
• dec
• double
• hex
• storedInteger
• storedIntegerToDouble
• oct
For example,
a = fi(pi);
n = storedInteger(a)
n =
25736
h = hex(a)
h =
6488
a.hex
ans =
6488
2-26
3
Fixed-Point Topics
Notation
Setup
This example may use display settings or preferences that are different from
what you are currently using. To ensure that your current display settings
and preferences are not changed by running this example, the example
automatically saves and restores them. The following code captures the
current states for any display settings or properties that the example changes.
For example, the following creates fi objects a and b with attributes shown in
the display, all of which we can specify when the variables are constructed.
Note that when the FractionLength property is not specified, it is set
automatically to "best precision" for the given word length, keeping the
most-significant bits of the value. When the WordLength property is not
specified it defaults to 16 bits.
3-2
Set Up Fixed-Point Objects
a = fi(pi)
a =
3.1416015625
b = fi(0.1)
b =
0.0999984741210938
The second and third numeric arguments specify Signed (true or 1 = signed,
false or 0 = unsigned), and WordLength in bits, respectively.
% Signed 8-bit
a = fi(pi, 1, 8)
a =
3.15625
3-3
3 Fixed-Point Topics
FractionLength: 5
a1 = sfi(pi,8)
a1 =
3.15625
% Unsigned 20-bit
b = fi(exp(1), 0, 20)
b =
2.71828079223633
b1 = ufi(exp(1), 20)
b1 =
2.71828079223633
3-4
Set Up Fixed-Point Objects
WordLength: 20
FractionLength: 18
Precision
a = ufi(0.1, 100);
bin(a)
ans =
110011001100110011001100110011001100110011001100110100000000000000000000000
Note that the infinite repeating binary expansion of 0.1 gets cut off
at the 52nd bit (in fact, the 53rd bit is significant and it is rounded
up into the 52nd bit). This is because double-precision floating-point
variables (the default MATLAB® data type), are stored in 64-bit
floating-point format, with 1 bit for the sign, 11 bits for the exponent,
and 52 bits for the mantissa plus one "hidden" bit for an effective 53
bits of precision. Even though double-precision floating-point has a very
large range, its precision is limited to 53 bits. For more information
on floating-point arithmetic, refer to Chapter 1 of Cleve Moler’s book,
Numerical Computing with MATLAB. The pdf version can be found here:
http://www.mathworks.com/company/aboutus/founders/clevemoler.html
So, why have more precision than floating-point? Because most fixed-point
processors have data stored in a smaller precision, and then compute with
larger precisions. For example, let’s initialize a 40-bit unsigned fi and
multiply using full-precision for products.
3-5
3 Fixed-Point Topics
a = fi(0.1, 0, 40);
bin(a)
ans =
1100110011001100110011001100110011001101
b = a*a
b =
0.0100000000000045
bin(b)
ans =
101000111101011100001010001111010111000011110101110000101000111101011100001
Access to Data
The data can be accessed in a number of ways which map to built-in data
types and binary strings. For example,
DOUBLE(A)
a = fi(pi);
double(a)
3-6
Set Up Fixed-Point Objects
ans =
3.1416015625
A.DOUBLE = ...
a.double = exp(1)
a =
2.71826171875
STOREDINTEGER(A)
storedInteger(a)
ans =
22268
returns the "stored integer" in the smallest built-in integer type available,
up to 64 bits.
3-7
3 Fixed-Point Topics
where
and
return the stored integer in binary, octal, unsigned decimal, and hexadecimal
strings, respectively.
bin(a)
ans =
0101011011111100
oct(a)
ans =
053374
3-8
Set Up Fixed-Point Objects
dec(a)
ans =
22268
hex(a)
ans =
56fc
set the stored integer from binary, octal, unsigned decimal, and hexadecimal
strings, respectively.
a.bin = '0110010010001000'
a =
3.1416015625
a.oct = '031707'
3-9
3 Fixed-Point Topics
a =
1.6180419921875
a.dec = '22268'
a =
2.71826171875
a.hex = '0333'
a =
0.0999755859375
Specifying FractionLength
3-10
Set Up Fixed-Point Objects
a = sfi(10,16,0)
a =
10
b = sfi(10,16)
b =
10
Note that the stored integer values of a and b are different, even though their
real-world values are the same. This is because the real-world value of a is
the stored integer scaled by 2^0 = 1, while the real-world value of b is the
stored integer scaled by 2^-11 = 0.00048828125.
storedInteger(a)
3-11
3 Fixed-Point Topics
ans =
10
storedInteger(b)
ans =
20480
Thus far, we have been specifying the numeric type properties by passing
numeric arguments to the fi constructor. We can also specify properties
by giving the name of the property as a string followed by the value of the
property:
a = fi(pi,'WordLength',20)
a =
3.14159393310547
help fi
or
doc fi
3-12
Set Up Fixed-Point Objects
T = numerictype
T =
The numeric type properties can be modified either when the object is created
by passing in parameter/value arguments
T = numerictype('WordLength',40,'FractionLength',37)
T =
T.Signed = false
T =
3-13
3 Fixed-Point Topics
Signedness: Unsigned
WordLength: 40
FractionLength: 37
a = fi(pi,'numerictype',T)
a =
3.14159265359194
b = fi(exp(1),'numerictype',T)
b =
2.71828182845638
a1 = fi(pi,T)
a1 =
3.14159265359194
3-14
Set Up Fixed-Point Objects
help numerictype
or
doc numerictype
Display Preferences
The display preferences for fi can be set with the fipref object. They can be
saved between MATLAB sessions with the savefipref command.
a = sfi(1,8,7)
a =
0.9921875
3-15
3 Fixed-Point Topics
bin(a)
ans =
01111111
while a 100-bit fractional number cannot (1 is displayed, when the exact value
is 1 - 2^-99):
b = sfi(1,100,99)
b =
bin(b)
ans =
011111111111111111111111111111111111111111111111111111111111111111111111111
format long g
3-16
Set Up Fixed-Point Objects
There are also other display options to make a more shorthand display of the
numeric type properties, and options to control the display of the value (as
real-world value, binary, octal, decimal integer, or hex).
help fipref
help savefipref
help format
or
doc fipref
doc savefipref
doc format
Cleanup
The following code sets any display settings or preferences that the example
changed back to their original states.
3-17
3 Fixed-Point Topics
Unsigned Integers.
be the binary digits of an n-bit unsigned integer, where each b(i) is either
one or zero. Then the value of b is
For example, let’s define a 3-bit unsigned integer quantizer, and enumerate
its range.
q = quantizer('ufixed',[3 0]);
[a,b] = range(q);
u = (a:eps(q):b)'
u =
0
1
2
3
3-18
View Fixed-Point Number Circles
4
5
6
7
b =
000
001
010
011
100
101
110
111
Let’s array them around a clock face with their corresponding binary and
decimal values.
fidemo.numbercircle(q);
Unsigned Fixed-Point.
Unsigned fixed-point values are unsigned integers that are scaled by a power
of two. We call the negative exponent of the power of two the "fractionlength".
uf = u*2^-f
q = quantizer('ufixed',[3 1]);
3-19
3 Fixed-Point Topics
[a,b] = range(q);
uf = (a:eps(q):b)'
uf =
0
0.5000
1.0000
1.5000
2.0000
2.5000
3.0000
3.5000
b =
000
001
010
011
100
101
110
111
Let’s array them around a clock face with their corresponding binary and
decimal values.
fidemo.numbercircle(q);
3-20
View Fixed-Point Number Circles
uf = u*2^-f
q = quantizer('ufixed',[3 3]);
[a,b] = range(q);
uf = (a:eps(q):b)'
uf =
0
0.1250
0.2500
0.3750
0.5000
0.6250
0.7500
0.8750
b =
000
3-21
3 Fixed-Point Topics
001
010
011
100
101
110
111
Let’s array them around a clock face with their corresponding binary and
decimal values.
fidemo.numbercircle(q);
be the binary digits of an n-bit signed integer, where each b(i) is either one or
zero. Then the value of b is
Note that the difference between this and the unsigned number is the
negative weight on the most-significant-bit (MSB).
For example, let’s define a 3-bit signed integer quantizer, and enumerate
its range.
q = quantizer('fixed',[3 0]);
[a,b] = range(q);
s = (a:eps(q):b)'
3-22
View Fixed-Point Number Circles
b = num2bin(q,s)
s =
-4
-3
-2
-1
0
1
2
3
b =
100
101
110
111
000
001
010
011
Let’s array them around a clock face with their corresponding binary and
decimal values.
The reason for this ungainly looking definition of negative numbers is that
addition of all numbers, both positive and negative, is carried out as if they
were all positive, and then the n+1 carry bit is discarded. The result will be
correct if there is no overflow.
3-23
3 Fixed-Point Topics
fidemo.numbercircle(q);
Signed Fixed-Point.
Signed fixed-point values are signed integers that are scaled by a power of
two. We call the negative exponent of the power of two the "fractionlength".
sf = s*2^-f
q = quantizer('fixed',[3 1]);
[a,b] = range(q);
sf = (a:eps(q):b)'
sf =
-2.0000
-1.5000
-1.0000
-0.5000
0
0.5000
1.0000
1.5000
b =
100
101
3-24
View Fixed-Point Number Circles
110
111
000
001
010
011
Let’s array them around a clock face with their corresponding binary and
decimal values.
fidemo.numbercircle(q);
sf = s*2^-f
For example, let’s define a 3-bit signed fractional fixed-point quantizer, which
implies a fractionlength of 2.
q = quantizer('fixed',[3 2]);
[a,b] = range(q);
sf = (a:eps(q):b)'
3-25
3 Fixed-Point Topics
sf =
-1.0000
-0.7500
-0.5000
-0.2500
0
0.2500
0.5000
0.7500
b =
100
101
110
111
000
001
010
011
Let’s array them around a clock face with their corresponding binary and
decimal values.
fidemo.numbercircle(q);
3-26
Perform Binary-Point Scaling
FI Construction
If s is true (signed) the leading or most significant bit (MSB) in the resulting
fi is always the sign bit.
For example, create a signed 8-bit long fi with a value of 0.5 and a scaling
of 2^(-7):
a = fi(0.5,true,8,7)
a =
0.5000
The fraction length or the scaling determines the position of the binary point
in the fi object.
The Fraction Length is Positive and Less than the Word Length
When the fraction length f is positive and less than the word length, the
binary point lies f places to the left of the least significant bit (LSB) and
within the word.
3-27
3 Fixed-Point Topics
For example, in a signed 3-bit fi with fraction length of 1 and value -0.5, the
binary point lies 1 place to the left of the LSB. In this case each bit is set to 1
and the binary equivalent of the fi with its binary point is 11.1 .
The real world value of -0.5 is obtained by multiplying each bit by its scaling
factor, starting with the LSB and working up to the signed MSB.
a = fi(-0.5,true,3,1)
bin(a)
storedInteger(a)
a =
-0.5000
ans =
111
ans =
-1
The Fraction Length is Positive and Greater than the Word Length
3-28
Perform Binary-Point Scaling
When the fraction length f is positive and greater than the word length, the
binary point lies f places to the left of the LSB and outside the word.
For example the binary equivalent of a signed 3-bit word with fraction length
of 4 and value of -0.0625 is ._111 Here _ in the ._111 denotes an unused
bit that is not a part of the 3-bit word. The first 1 after the _ is the MSB or
the sign bit.
b = fi(-0.0625,true,3,4)
bin(b)
storedInteger(b)
b =
-0.0625
ans =
111
ans =
-1
The Fraction Length is a Negative Integer and Less than the Word Length
3-29
3 Fixed-Point Topics
When the fraction length f is negative the binary point lies f places to the
right of LSB and is outside the physical word.
The real world value of -4 is obtained by multiplying each bit by its scaling
factor 2^(-f), i.e. 2(-(-2)) = 2^(2) for the LSB, and then adding the
products together.
bin(c) and storedInteger(c) will still give 111 and -1 as in the previous
two examples.
c = fi(-4,true,3,-2)
bin(c)
storedInteger(c)
c =
-4
ans =
111
ans =
-1
3-30
Perform Binary-Point Scaling
The Fraction Length is Set Automatically to the Best Precision Possible and is
Negative
In this example we create a signed 3-bit fi where the fraction length is set
automatically depending on the value that the fi is supposed to contain.
The resulting fi has a value of 6, with a wordlength of 3 bits and a fraction
length of -1. Here the binary point is 1 place to the right of the LSB: 011_..
The _ is again an unused bit and the first 1 before the _ is the LSB. The
leading 1 is the sign bit.
d = fi(5,true,3)
bin(d)
storedInteger(d)
d =
ans =
011
ans =
3-31
3 Fixed-Point Topics
This is an interactive example that allows the user to change the fraction
length of a 3-bit fixed-point number by moving the binary point using a slider.
The fraction length can be varied from -3 to 5 and the user can change the
value of the 3 bits to ’0’ or ’1’ for either signed or unsigned numbers.
The "Scaling factors" above the 3 bits display the scaling or weight that each
bit is given for the specified signedness and fraction length. The fi code,
the double precision real-world value and the fixed-point attributes are also
displayed.
3-32
Develop Fixed-Point Algorithms
2) Instrument the code to visualize the dynamic range of the output and state.
Data-Type-Independent Algorithm
for k=1:length(x)
3-33
3 Fixed-Point Topics
% Reset states
z = [0;0];
hscope1 = NumericTypeScope;
hscope2 = NumericTypeScope;
for k=1:length(x)
y(k) = b(1)*x(k) + z(1);
z(1) = (b(2)*x(k) + z(2)) - a(2)*y(k);
z(2) = b(3)*x(k) - a(3)*y(k);
% process the data and update the visual.
step(hscope1,z);
end
step(hscope2,y);
3-34
Develop Fixed-Point Algorithms
Let us first analyze the information displayed for variable z (state). From the
histogram we can see that the dynamic range lies between ( ].
By default, the scope uses a word length of 16 bits with zero tolerable
overflows. This results in a data type of numerictype(true,16, 14) since we
need at least 2 integer bit to avoid overflows. You can get more information
on the statistical data from the Input Data and Resulting Type panels. From
the Input Data panel we can see that the data has both positive and negative
values and hence a signed quantity which is reflected in the suggested
numerictype. Also, the maximum data value is 1.51 which can be represented
by the suggested type.
Next, let us look at variable y (output). From the histogram plot we see that
the dynamic range lies between ( ].
By default, the scope uses a word length of 16 bits with zero tolerable
overflows. This results in a data type of numerictype(true,16, 14) since we
need at least 2 integer bits to avoid overflows. With this suggested type you
see no overflows or underflows.
We convert variables to fixed-point and run the algorithm again. We will turn
on logging to see the overflows and underflows introduced by the selected
data types.
3-35
3 Fixed-Point Topics
for k=1:length(x)
y(k) = b(1)*x(k) + z(1);
z(1) = (b(2)*x(k) + z(2)) - a(2)*y(k);
z(2) = b(3)*x(k) - a(3)*y(k);
end
% Reset the logging mode.
fp.LoggingMode = default_loggingmode;
In this example, we have redefined the fixed-point variables with the same
names as the floating-point so that we could inline the algorithm code for
clarity. However, it is a better practice to enclose the algorithm code in a
MATLAB® file function that could be called with either floating-point or
fixed-point variables. See filimitcycledemo.m for an example of writing and
using a datatype-agnostic algorithm.
We will now plot the magnitude response of the floating-point and fixed-point
results and the response of the filter to see if the filter behaves as expected
when it is converted to fixed-point.
n = length(x);
f = linspace(0,0.5,n/2);
x_response = 20*log10(abs(fft(double(x))));
ydouble_response = 20*log10(abs(fft(ydouble)));
y_response = 20*log10(abs(fft(double(y))));
plot(f,x_response(1:n/2),'c-',...
f,ydouble_response(1:n/2),'bo-',...
f,y_response(1:n/2),'gs-');
ylabel('Magnitude in dB');
xlabel('Normalized Frequency');
legend('Input','Floating point output','Fixed point output','Location','Bes
title('Magnitude response of Floating-point and Fixed-point results');
3-36
Develop Fixed-Point Algorithms
h = fft(double(b),n)./fft(double(a),n);
h = h(1:end/2);
clf
hax = axes;
plot(hax,f,20*log10(abs(h)));
set(hax,'YLim',[-40 0]);
title('Magnitude response of the filter');
ylabel('Magnitude in dB')
xlabel('Frequency');
Notice that the high frequencies in the input signal are attenuated by the
low-pass filter which is the expected behavior.
clf
n = (0:length(y)-1)';
e = double(lsb(y));
plot(n,double(y)-ydouble,'.-r', ...
[n(1) n(end)],[e/2 e/2],'c', ...
[n(1) n(end)],[-e/2 -e/2],'c')
text(n(end),e/2,'+1/2 LSB','HorizontalAlignment','right','VerticalAlignment
text(n(end),-e/2,'-1/2 LSB','HorizontalAlignment','right','VerticalAlignmen
xlabel('n (samples)'); ylabel('error')
Simulink®
If you have Simulink® and Fixed-Point Designer™, you can run this model,
which is the equivalent of the algorithm above. The output, y_sim is a
fixed-point variable equal to the variable y calculated above in MATLAB code.
3-37
3 Fixed-Point Topics
MATLAB code in the example above. Double-click on the blocks to see the
settings.
if fidemo.hasSimulinkLicense
% Verify that the Simulink results are the same as the MATLAB file
isequal(y, out_sim.get('y_sim'))
end
ans =
In order to simplify the example, we have taken the default math parameters:
round-to-nearest, saturate on overflow, full precision products and sums. We
can modify all of these parameters to match an actual system.
3-38
Develop Fixed-Point Algorithms
different input? See the help for fi, fimath, and numerictype for information
on how to set other parameters, such as rounding mode, and overflow mode.
3-39
3 Fixed-Point Topics
Introduction
You can use the CORDIC rotation computing mode to calculate sine and
cosine, and also polar-to-cartesian conversion operations. In this mode, the
vector magnitude and an angle of rotation are known and the coordinate (X-Y)
components are computed after rotation.
3-40
Calculate Fixed-Point Sine and Cosine
Where:
Introduction
The cordicsincos function calculates the sine and cosine of input angles
in the range [-2*pi 2*pi) using the CORDIC algorithm. This function takes
an angle (radians) and the number of iterations as input arguments. The
function returns approximations of sine and cosine.
The CORDIC computation outputs are scaled by the rotator gain. This gain is
accounted for by pre-scaling the initial constant value.
Initialization
3-41
3 Fixed-Point Topics
The judicious choice of initial values allows the algorithm to directly compute
both sine and cosine simultaneously. After iterations, these initial values
lead to the following outputs as approaches :
The MATLAB code for the CORDIC algorithm (rotation mode) kernel portion
is as follows (for the case of scalar x, y, and z). This same code is used for both
fixed-point and floating-point operations:
3-42
Calculate Fixed-Point Sine and Cosine
Example
NITERS ERROR
3-43
3 Fixed-Point Topics
------ ----------
1 0.10191021
2 0.13966630
3 0.03464449
4 0.03846157
5 0.00020393
6 0.01776952
7 0.00888037
8 0.00436052
9 0.00208192
10 0.00093798
figure(1); clf;
bar(1:niters, cos_err(1:niters));
xlabel('Number of iterations','fontsize',12,'fontweight','b');
ylabel('Error','fontsize',12,'fontweight','b');
title('CORDIC approximation error for cos(pi/5) computation',...
'fontsize',12,'fontweight','b');
axis([0 niters 0 0.14]);
Niter2Draw = 5;
figure(2), clf, hold on
plot(cos(0:0.1:pi/2), sin(0:0.1:pi/2), 'b--'); % semi-circle
for i=1:Niter2Draw
plot([0 x_cos(i)],[0 y_sin(i)], 'LineWidth', 2); % CORDIC iteration res
text(x_cos(i),y_sin(i),int2str(i),'fontsize',12,'fontweight','b');
end
plot(cos(theta), sin(theta), 'r*', 'MarkerSize', 20); % IDEAL result
xlabel('X (COS)','fontsize',12,'fontweight','b')
ylabel('Y (SIN)','fontsize',12,'fontweight','b')
title('CORDIC iterations for cos(pi/5) computation',...
'fontsize',12,'fontweight','b')
axis equal;
3-44
Calculate Fixed-Point Sine and Cosine
axis square;
stepSize = pi/256;
thRadDbl = (-2*pi):stepSize:(2*pi - stepSize);
thRadFxp = sfi(thRadDbl, 12); % signed, 12-bit fixed-point values
sinThRef = sin(double(thRadFxp)); % reference results
Use 12-bit quantized inputs and vary number of iterations from 4 to 10.
3-45
3 Fixed-Point Topics
figure;
fracLen = cdcSinTh.FractionLength;
plot(thRadFxp, abs(errCdcRef) * pow2(fracLen));
set(gca,'XTick',-2*pi:pi/2:2*pi);
set(gca,'XTickLabel',...
{'-2*pi', '-3*pi/2', '-pi', '-pi/2', ...
'0', 'pi/2', 'pi', '3*pi/2','2*pi'});
ylabel(sprintf('LSB Error: 1 LSB = 2^{-%d}',fracLen),'fontsize',12,'fontwei
title('LSB Error: 12-bit CORDICSIN; N=10','fontsize',12,'fontweight','b');
axis([-2*pi 2*pi 0 6]);
fft_mag = abs(fft(double(cdcSinTh)));
max_mag = max(fft_mag);
mag_db = 20*log10(fft_mag/max_mag);
figure;
hold on;
plot(0:1023, mag_db);
plot(0:1023, zeros(1,1024),'r--'); % Normalized peak (0 dB)
plot(0:1023, -62.*ones(1,1024),'r--'); % Noise floor level
ylabel('dB Magnitude','fontsize',12,'fontweight','b');
title('62 dB Noise Floor: 12-bit CORDICSIN; N=10',...
'fontsize',12,'fontweight','b');
% axis([0 1023 -120 0]); full FFT
axis([0 round(1024*(pi/8)) -100 10]); % zoom in
set(gca,'XTick',[0 round(1024*pi/16) round(1024*pi/8)]);
set(gca,'XTickLabel',{'0','pi/16','pi/8'});
3-46
Calculate Fixed-Point Sine and Cosine
You can generate a MEX function from MATLAB code using the MATLAB®
fiaccel function. Typically, running a generated MEX function can improve
the simulation speed, although the actual speed improvement depends on
the simulation platform being used. The following example shows how to
accelerate the fixed-point cordicsincos function using fiaccel.
The fiaccel function compiles the MATLAB code into a MEX function. This
step requires the creation of a temporary directory and write permissions
in this directory.
tempdirObj = fidemo.fiTempdir('fi_sin_cos_demo');
When you declare the number of iterations to be a constant (e.g., 10) using
coder.newtype('constant',10), the compiled angle look-up table will also
be constant, and thus won’t be computed at each iteration. Also, when you
call cordicsincos_mex, you will not need to give it the input argument
for the number of iterations. If you pass in the number of iterations, the
MEX-function will error.
The data type of the input parameters determines whether the cordicsincos
function performs fixed-point or floating-point calculations. When MATLAB
generates code for this file, code is only generated for the specific data
type. For example, if the THETA input argument is fixed point, then only
fixed-point code is generated.
tstart = tic;
cordicsincos(thRadFxp,10);
telapsed_Mcordicsincos = toc(tstart);
3-47
3 Fixed-Point Topics
Now, compare the speed. Type the following at the MATLAB command line
to see the speed improvement on your platform:
fiaccel_speedup = telapsed_Mcordicsincos/telapsed_MEXcordicsincos;
clear cordicsincos_mex;
status = tempdirObj.cleanUp;
The sin and cos methods of the fi object in the Fixed-Point Designer
approximate the MATLAB® builtin floating-point sin and cos functions,
using a lookup table-based approach with simple nearest-neighbor linear
interpolation between values. This approach allows for a small real-valued
lookup table and uses simple arithmetic.
Two important design considerations of a lookup table are its size and its
accuracy. It is not possible to create a table for every possible input value .
It is also not possible to be perfectly accurate due to the quantization of or
lookup table values.
3-48
Calculate Fixed-Point Sine and Cosine
tblValsNT = numerictype(1,16,15);
quarterSinDblFltPtVals = (sin(2*pi*((0:63) ./ 256)))';
endpointQuantized_Plus1 = 1.0 - double(eps(fi(0,tblValsNT)));
halfSinWaveDblFltPtVals = ...
[quarterSinDblFltPtVals; ...
endpointQuantized_Plus1; ...
flipud(quarterSinDblFltPtVals(2:end))];
fullSinWaveDblFltPtVals = ...
[halfSinWaveDblFltPtVals; -halfSinWaveDblFltPtVals];
3-49
3 Fixed-Point Topics
Next, we get the 16-bit stored unsigned integer value from this in-range
fixed-point FI angle value:
normConst_NT = numerictype(0,32,31);
normConstant = fi(65536/51472, normConst_NT);
fullScaleIdx = normConstant * idxUFIX16;
idxUFIX16(:) = fullScaleIdx;
The top 8 most significant bits (MSBs) of this full-scale unsigned 16-bit
index idxUFIX16 are used to directly index into the 8-bit sine lookup table.
Two table lookups are performed, one at the computed table index location
lutValBelow, and one at the next index location lutValAbove:
A real multiply is used to determine the weighted difference between the two
points. This results in a simple calculation (equivalent to one product and two
sums) to obtain the interpolated fixed-point sine result:
3-50
Calculate Fixed-Point Sine and Cosine
Example
Using the above algorithm, here is an example of the lookup table and linear
interpolation process used to compute the value of SIN for a fixed-point input
inpValInRange = 0.425 radians:
% Do two table lookups using unsigned 8-bit integer index (i.e., 8 MSBs)
idxUint8MSBs = storedInteger(bitsliceget(idxUFIX16, 16, 9));
zeroBasedIdx = int16(idxUint8MSBs); % zero-based table index valu
lutValBelow = FI_SIN_LUT(zeroBasedIdx + 1); % 1st table lookup value
lutValAbove = FI_SIN_LUT(zeroBasedIdx + 2); % 2nd table lookup value
x_vals = 0:(pi/128):(pi/4);
xIdxLo = zeroBasedIdx - 1;
xIdxHi = zeroBasedIdx + 4;
figure; hold on; axis([x_vals(xIdxLo) x_vals(xIdxHi) 0.25 0.65]);
plot(x_vals(xIdxLo:xIdxHi), double(FI_SIN_LUT(xIdxLo:xIdxHi)), 'b^--');
plot([x_vals(zeroBasedIdx+1) x_vals(zeroBasedIdx+2)], ...
3-51
3 Fixed-Point Topics
stepSize = pi/256;
thRadDbl = (-2*pi):stepSize:(2*pi - stepSize); % double precision floating-
thRadFxp = sfi(thRadDbl, 12); % signed, 12-bit fixed-point inputs
3-52
Calculate Fixed-Point Sine and Cosine
figure;
fracLen = fxpSinTh.FractionLength;
plot(thRadFxp, abs(errSinRef) * pow2(fracLen));
set(gca,'XTick',-2*pi:pi/2:2*pi);
set(gca,'XTickLabel',...
{'-2*pi', '-3*pi/2', '-pi', '-pi/2', ...
'0', 'pi/2', 'pi', '3*pi/2','2*pi'});
ylabel(sprintf('LSB Error: 1 LSB = 2^{-%d}',fracLen),'fontsize',12,'fontwei
title('LSB Error: 16-bit Fixed-Point SIN with 12-bit Inputs','fontsize',12,
axis([-2*pi 2*pi 0 8]);
fft_mag = abs(fft(double(fxpSinTh)));
max_mag = max(fft_mag);
mag_db = 20*log10(fft_mag/max_mag);
figure;
hold on;
plot(0:1023, mag_db);
plot(0:1023, zeros(1,1024),'r--'); % Normalized peak (0 dB)
plot(0:1023, -64.*ones(1,1024),'r--'); % Noise floor level (dB)
ylabel('dB Magnitude','fontsize',12,'fontweight','b');
title('64 dB Noise Floor: 16-bit Fixed-Point SIN with 12-bit Inputs',...
'fontsize',12,'fontweight','b');
% axis([0 1023 -120 0]); full FFT
axis([0 round(1024*(pi/8)) -100 10]); % zoom in
set(gca,'XTick',[0 round(1024*pi/16) round(1024*pi/8)]);
set(gca,'XTickLabel',{'0','pi/16','pi/8'});
3-53
3 Fixed-Point Topics
• 2 table lookups
• 1 multiplication
• 2 additions
References
3-54
Calculate Fixed-Point Arctangent
Introduction
3-55
3 Fixed-Point Topics
As approaches :
Introduction
Initialization
3-56
Calculate Fixed-Point Arctangent
The MATLAB code for the CORDIC algorithm (vectoring mode) kernel portion
is as follows (for the case of scalar x, y, and z). This same code is used for both
fixed-point and floating-point operations:
Example
3-57
3 Fixed-Point Topics
The following output shows the CORDIC angle accumulation (in degrees)
through 10 iterations. Note that the 5th iteration produced less error than
the 6th iteration, and that the calculated angle quickly converges to the
actual input angle afterward.
3-58
Calculate Fixed-Point Arctangent
y_n = yf(end)
y_n =
-0.0018
x_n = xf(end)
x_n =
1.6468
3-59
3 Fixed-Point Topics
figno = 1;
fidemo.fixpt_atan2_demo_plot(figno, xf, yf) %Vectoring Mode CORDIC Iteratio
1 The algorithmic error that results from the CORDIC rotation angle being
represented by a finite number of basic angles.
2 The quantization or rounding error that results from the finite precision
representation of the angle lookup table, and from the finite precision
arithmetic used in fixed-point operations.
format long
cordic_algErr_real_world_value = max(abs((atan2(inYflt, inXflt) - zflt)))
cordic_algErr_real_world_value =
4.753112306290497e-04
3-60
Calculate Fixed-Point Arctangent
The log base 2 error is related to the number of iterations. In this example,
we use 12 iterations (i.e., accurate to 11 binary digits), so the magnitude of
the error is less than
cordic_algErr_bits = log2(cordic_algErr_real_world_value)
cordic_algErr_bits =
-11.038839889583048
Once the quantization error dominates the overall error, i.e., the quantization
error is greater than the algorithmic error, increasing the total number of
iterations won’t significantly decrease the overall error of the fixed-point
CORDIC algorithm. You should pick your fraction lengths and total number
of iterations to ensure that the quantization error is smaller than the
algorithmic error. In the CORDIC algorithm, the precision increases by one
bit every iteration. Thus, there is no reason to pick a number of iterations
greater than the precision of the input data.
x(:) = x0 - bitsra(y,i);
y(:) = y + bitsra(x0,i);
To measure the error from the fixed-point algorithm, and not the differences
in input values, compute the floating-point reference with the same inputs as
the fixed-point CORDIC algorithm.
3-61
3 Fixed-Point Topics
The error depends on the number of iterations and the precision of the input
data. In the above example, the input data is in the range [-1, +1], and the
fraction length is 14. From the following tables showing the maximum error
at each iteration, and the figure showing the overall error of the CORDIC
algorithm, you can see that the error decreases by about 1 bit per iteration
until the precision of the data is reached.
max_cordicErr_bits = log2(max_cordicErr_real_world_value);
fprintf('Iterations: %2d, Max error in bits: %g\n',[iterations; max_cordicE
figno = figno + 1;
fidemo.fixpt_atan2_demo_plot(figno, theta, cordic_err)
3-62
Calculate Fixed-Point Arctangent
You can generate a MEX function from MATLAB code using the MATLAB®
fiaccel command. Typically, running a generated MEX function can improve
the simulation speed, although the actual speed improvement depends on
the simulation platform being used. The following example shows how to
accelerate the fixed-point cordicatan2 algorithm using fiaccel.
The fiaccel function compiles the MATLAB code into a MEX function. This
step requires the creation of a temporary directory and write permissions
in that directory.
tempdirObj = fidemo.fiTempdir('fixpt_atan2_demo');
When you declare the number of iterations to be a constant (e.g., 12) using
coder.newtype('constant',12), the compiled angle lookup table will also
be constant, and thus won’t be computed at each iteration. Also, when you
call the compiled MEX file cordicatan2_mex, you will not need to give it the
input argument for the number of iterations. If you pass in the number of
iterations, the MEX function will error.
The data type of the input parameters determines whether the cordicatan2
function performs fixed-point or floating-point calculations. When MATLAB
generates code for this file, code is only generated for the specific data type.
For example, if the inputs are fixed point, only fixed-point code is generated.
tstart = tic;
cordicatan2(inYfix,inXfix,Niter);
telapsed_Mcordicatan2 = toc(tstart);
3-63
3 Fixed-Point Topics
Now, compare the speed. Type the following in the MATLAB command
window to see the speed improvement on your specific platform:
fiaccel_speedup = telapsed_Mcordicatan2/telapsed_MEXcordicatan2;
clear cordicatan2_mex;
status = tempdirObj.cleanUp;
The approximation of arctangent over the interval of [-1, +1] using the
Chebyshev polynomial of the first kind is summarized in the following
formula:
where
3-64
Calculate Fixed-Point Arctangent
You can obtain four quadrant output through angle correction based on the
properties of the arctangent function.
3-65
3 Fixed-Point Topics
The log base 2 error shows the number of binary digits of accuracy. The 12th
iteration of the CORDIC algorithm has an estimated angle accuracy of :
max_cordicAlgErr_bits = log2(max_cordicAlgErr);
fprintf('Iterations: %2d, CORDIC algorithmic error in bits: %g\n',...
[[8,12]; max_cordicAlgErr_bits(:)']);
The following code shows the magnitude of the maximum algorithmic error of
the polynomial approximation for orders 3, 5, and 7:
3-66
Calculate Fixed-Point Arctangent
max_polyAlgErr = max(abs(poly_algErr'));
fprintf('Order: %d, Polynomial approximation algorithmic error in real-worl
[3:2:7; max_polyAlgErr(:)']);
The log base 2 error shows the number of binary digits of accuracy.
max_polyAlgErr_bits = log2(max_polyAlgErr);
fprintf('Order: %d, Polynomial approximation algorithmic error in bits: %g\
[3:2:7; max_polyAlgErr_bits(:)']);
figno = figno + 1;
fidemo.fixpt_atan2_demo_plot(figno, theta, cordic_algErr, poly_algErr)
Assume the input and output word lengths are constrained to 16 bits by
the hardware, and the 5th order Chebyshev polynomial is used in the
approximation. Because the dynamic range of inputs x, y and y/x are all
within [-1, +1], you can avoid overflow by picking a signed fixed-point input
data type with a word length of 16 bits and a fraction length of 14 bits. The
coefficients of the polynomial are purely fractional and within (-1, +1), so
we can pick their data types as signed fixed point with a word length of 16
bits and a fraction length of 15 bits (best precision). The algorithm is robust
because is within [-1, +1], and the multiplication of the coefficients and
is within (-1, +1). Thus, the dynamic range will not grow, and due to the
pre-determined fixed-point data types, overflow is not expected.
3-67
3 Fixed-Point Topics
function z = chebyPoly_atan_fltpt(y,x,N,constA,Tz,RoundingMethodStr)
tmp = y/x;
switch N
case 3
z = constA(1)*tmp + constA(2)*tmp^3;
case 5
z = constA(1)*tmp + constA(2)*tmp^3 + constA(3)*tmp^5;
case 7
z = constA(1)*tmp + constA(2)*tmp^3 + constA(3)*tmp^5 + constA(4
otherwise
disp('Supported order of Chebyshev polynomials are 3, 5 and 7');
end
function z = chebyPoly_atan_fixpt(y,x,N,constA,Tz,RoundingMethodStr)
if (N == 5) || (N == 7)
tmp5 = fi(0, 'numerictype',Tx, 'RoundingMethod', RoundingMethodStr);
3-68
Calculate Fixed-Point Arctangent
if N == 7
tmp7 = fi(0, 'numerictype',Tx, 'RoundingMethod', RoundingMethodS
tmp7(:) = tmp5 * tmp2; % (y/x)^7
z(:) = z + constA(4)*tmp7; %for order N = 7
end
end
function z = poly_atan2(y,x,N,constA,Tz,RoundingMethodStr)
if nargin < 5
% floating-point algorithm
fhandle = @chebyPoly_atan_fltpt;
Tz = [];
RoundingMethodStr = [];
z = zeros(size(y));
else
% fixed-point algorithm
fhandle = @chebyPoly_atan_fixpt;
%pre-allocate output
z = fi(zeros(size(y)), 'numerictype', Tz, 'RoundingMethod', Rounding
end
if x(idx) < 0
% second and third quadrant
3-69
3 Fixed-Point Topics
if y(idx) < 0
z(idx) = -pi + z(idx);
else
z(idx) = pi - z(idx);
end
else % fourth quadrant
if y(idx) < 0
z(idx) = -z(idx);
end
end
end
3-70
Calculate Fixed-Point Arctangent
zfix5p = fidemo.poly_atan2(inYfix,inXfix,ord(2),constAfix5,Tz,'Floor'); % 5
zfix7p = fidemo.poly_atan2(inYfix,inXfix,ord(3),constAfix7,Tz,'Floor'); % 7
poly_quantErr = bsxfun(@minus, [zfltp3;zfltp5;zfltp7], double([zfix3p;zfix5
max_polyQuantErr_real_world_value = max(abs(poly_quantErr'));
max_polyQuantErr_bits = log2(max_polyQuantErr_real_world_value);
fprintf('PolyOrder: %2d, Quant error in bits: %g\n',...
[ord; max_polyQuantErr_bits]);
figno = figno + 1;
fidemo.fixpt_atan2_demo_plot(figno, theta, poly_err)
3-71
3 Fixed-Point Topics
poly5_quantErrFloor = max(abs(poly_quantErr(2,:)));
poly5_quantErrFloor_bits = log2(poly5_quantErrFloor)
poly5_quantErrFloor_bits =
-12.324996933210334
zfixp5n = fidemo.poly_atan2(inYfix,inXfix,5,constAfix5,Tz,'Nearest');
poly5_quantErrNearest = max(abs(zfltp5 - double(zfixp5n)));
poly5_quantErrNearest_bits = log2(poly5_quantErrNearest)
set(0, 'format', origFormat); % reset MATLAB output format
poly5_quantErrNearest_bits =
-13.175966487895451
There are many lookup table based approaches that may be used to implement
fixed-point argtangent approximations. The following is a low-cost approach
based on a single real-valued lookup table and simple nearest-neighbor linear
interpolation.
3-72
Calculate Fixed-Point Arctangent
Using a single real-valued lookup table simplifies the index computation and
the overall arithmetic required to achieve very good accuracy of the results.
These simplifications yield a relatively high speed performance as well as
relatively low memory requirements.
Two important design considerations of a lookup table are its size and its
accuracy. It is not possible to create a table for every possible input value.
It is also not possible to be perfectly accurate due to the quantization of the
lookup table values.
3-73
3 Fixed-Point Topics
The two lookup table values, with the remainder (rFraction) value, are used
to perform a simple nearest-neighbor linear interpolation. A real multiply
is used to determine the weighted difference between the two points. This
results in a simple calculation (equivalent to one product and two sums) to
obtain the interpolated fixed-point result:
Finally, based on the original signs and relative magnitudes of y and x, the
output result is formed using simple octant-correction logic and arithmetic.
The first-octant [0, pi/4] angle value results are added or subtracted with
constants to form the octant-corrected angle outputs.
3-74
Calculate Fixed-Point Arctangent
You can call the atan2 function directly using fixed-point or floating-point
inputs. The lookup table based algorithm is used for the fixed-point atan2
implementation:
zFxpLUT = atan2(inYfix,inXfix);
You can compute the overall error by comparing the fixed-point lookup table
based approximation to the builtin atan2 function. The ideal reference output
is zfltRef.
figno = figno + 1;
fidemo.fixpt_atan2_demo_plot(figno, theta, lut_err)
As was done previously, you can compute the overall error by comparing
the fixed-point approximation(s) to the builtin atan2 function. The ideal
reference output is zfltRef.
3-75
3 Fixed-Point Topics
• 1 division
• (N+1) multiplications
• (N-1)/2 additions
• 1 division
• 2 table lookups
• 1 multiplication
• 2 additions
References
3-76
Compute Sine and Cosine Using CORDIC Rotation Kernel
Introduction
You can use a CORDIC rotation computing mode algorithm to calculate sine
and cosine simultaneously, compute polar-to-cartesian conversions, and for
other operations. In the rotation mode, the vector magnitude and an angle
of rotation are known and the coordinate (X-Y) components are computed
after rotation.
3-77
3 Fixed-Point Topics
Where:
3-78
Compute Sine and Cosine Using CORDIC Rotation Kernel
ytmp = y;
for idx = 1:n
if z < 0
z(:) = accumpos(z, inpLUT(idx));
x(:) = accumpos(x, ytmp);
y(:) = accumneg(y, xtmp);
else
z(:) = accumneg(z, inpLUT(idx));
x(:) = accumneg(x, ytmp);
y(:) = accumpos(y, xtmp);
end
xtmp = bitsra(x, idx); % bit-shift-right for multiply by 2^(-idx)
ytmp = bitsra(y, idx); % bit-shift-right for multiply by 2^(-idx)
end
The judicious choice of initial values allows the CORDIC kernel rotation mode
algorithm to directly compute both sine and cosine simultaneously.
•
•
3-79
3 Fixed-Point Topics
Example
Suppose that you have a rotation angle sensor (e.g. in a servo motor) that
uses formatted integer values to represent measured angles of rotation. Also
suppose that you have a 16-bit integer arithmetic unit that can perform
add, subtract, shift, and memory operations. With such a device, you could
implement the CORDIC rotation kernel to efficiently compute cosine and sine
(equivalently, cartesian X and Y coordinates) from the sensor angle values,
without the use of multiplies or large lookup tables.
3-80
Compute Sine and Cosine Using CORDIC Rotation Kernel
References
3-81
3 Fixed-Point Topics
This example addresses the first case, where your target is fixed-point, and
you want an algorithm that is independent of data type because it is easier
to develop and debug.
In this example you will learn various coding methods that can be applied
across systems. The significant design patterns used in this example are
the following:
3-82
Perform QR Factorization Using CORDIC
Setup
So this example does not change your preferences or settings, we store the
original state here, and restore them at the end.
3-83
3 Fixed-Point Topics
R = A;
Q = coder.nullcopy(repmat(A(:,1),1,m)); % Declare type and size of Q
Q(:) = eye(m); % Initialize Q
for j=1:n
for i=j+1:m
[R(j,j:end),R(i,j:end),Q(:,j),Q(:,i)] = ...
cordicgivens(R(j,j:end),R(i,j:end),Q(:,j),Q(:,i),niter,Kn);
end
end
end
R = A;
Q = eye(m)
Hence, you need to declare the type and size of Q in one step, and then
initialize it in a second step. This gives MATLAB Coder the information it
needs to create an efficient C program with the correct types and sizes. In
the finished code you initialize output Q to be an M-by-M identity matrix and
the same data type as A, like this:
3-84
Perform QR Factorization Using CORDIC
You will notice this pattern of assignment again and again. This is another
key enabler of data-type independent code.
[R(j,j:end),R(i,j:end),Q(:,j),Q(:,i)] = ...
cordicgivens(R(j,j:end),R(i,j:end),Q(:,j),Q(:,i),niter,Kn);
3-85
3 Fixed-Point Topics
u0 = u;
if y(1)<0
% Counter-clockwise rotation
% x and y form R, u and v form Q
x(:) = x - bitsra(y, i); u(:) = u - bitsra(v, i);
y(:) = y + bitsra(x0,i); v(:) = v + bitsra(u0,i);
else
% Clockwise rotation
% x and y form R, u and v form Q
x(:) = x + bitsra(y, i); u(:) = u + bitsra(v, i);
y(:) = y - bitsra(x0,i); v(:) = v - bitsra(u0,i);
end
end
% Set y(1) to exactly zero so R will be upper triangular without round of
% showing up in the lower triangle.
y(1) = 0;
% Normalize the CORDIC gain
x(:) = Kn * x; u(:) = Kn * u;
y(:) = Kn * y; v(:) = Kn * v;
end
The bit shifts in each iteration are performed with the bit shift right
arithmetic (bitsra) function instead of bitshift, multiplication by 0.5, or
division by 2, because bitsra
3-86
Perform QR Factorization Using CORDIC
x = x + bitsra(y, i);
then the left-hand-side x takes on the type of the right-hand-side sum. This
programming style leads to changing the data type of x in fixed-point code,
and is discouraged.
This function returns the inverse of the CORDIC growth factor after niter
iterations. It is needed because CORDIC rotations grow the values by a factor
of approximately 1.6468, depending on the number of iterations, so the gain
is normalized in the last step of cordicgivens by a multiplication by the
inverse Kn = 1/1.6468 = 0.60725.
function Kn = inverse_cordic_growth_constant(niter)
Kn = 1/prod(sqrt(1+2.^(-2*(0:double(niter)-1))));
end
growth = prod(sqrt(1+2.^(-2*(0:double(niter)-1))))
inverse_growth = 1 ./ growth
3-87
3 Fixed-Point Topics
3-88
Perform QR Factorization Using CORDIC
32 1.646760258121065 0 0.607252935008881
The givensrotation function uses division and square root, which are
expensive in fixed-point, but good for floating-point algorithms.
Here is a 3-by-3 example that follows the CORDIC rotations through each
step of the algorithm. The algorithm uses orthogonal rotations to zero out
the subdiagonal elements of R using the diagonal elements as pivots. The
same rotations are applied to the identity matrix, thus producing orthogonal
Q such that Q*R = A.
3-89
3 Fixed-Point Topics
Q = [ 1 0 0
0 1 0
0 0 1]
The first rotation is about the first and second row of R and the first and
second column of Q. Element R(1,1) is the pivot and R(2,1) rotates to 0.
In the following plot, you can see the growth in x in each of the CORDIC
iterations. The growth is factored out at the last step by multiplying it by Kn
= 0.60725. You can see that y(1) iterates to 0. Initially, the point [x(1),
y(1)] is in the third quadrant, and is reflected into the first quadrant before
the start of the CORDIC iterations.
The second rotation is about the first and third row of R and the first and
third column of Q. Element R(1,1) is the pivot and R(3,1) rotates to 0.
3-90
Perform QR Factorization Using CORDIC
The third rotation is about the second and third row of R and the second and
third column of Q. Element R(2,2) is the pivot and R(3,2) rotates to 0.
R =
1.3434 0.1235 0.8954
0 0.7054 0.6308
0 0 0.2987
Q =
-0.6105 0.6134 0.5011
-0.5781 0.0875 -0.8113
-0.5415 -0.7849 0.3011
3-91
3 Fixed-Point Topics
0.0000 1.0000 0
0.0000 0 1.0000
You can see the error difference by subtracting the identity matrix.
You can verify that Q*R is close to A by subtracting to see the error difference.
Since Q is orthogonal, you know that all of its values are between -1 and +1.
In floating-point, there is no decision about the type of Q: it should be the
same floating-point type as A. However, in fixed-point, you can do better than
making Q have the identical fixed-point type as A. For example, if A has word
length 16 and fraction length 8, and if we make Q also have word length 16
and fraction length 8, then you force Q to be less accurate than it could be and
waste the upper half of the fixed-point range.
The best type for Q is to make it have full range of its possible outputs, plus
accommodate the 1.6468 CORDIC growth factor in intermediate calculations.
Therefore, assuming that the word length of Q is the same as the word length
of input A, then the best fraction length for Q is 2 bits less than the word
length (one bit for 1.6468 and one bit for the sign).
3-92
Perform QR Factorization Using CORDIC
Q = coder.nullcopy(repmat(A(:,1),1,m));
Q(:) = eye(m);
end
This is true because the each element of R is formed from orthogonal rotations
from its corresponding column in A, so the largest that any element R(i,j)
can get is if all of the elements of its corresponding column A(:,j) were
rotated to a single value. In other words, the largest possible value will be
bounded by the 2-norm of A(:,j). Since the 2-norm of A(:,j) is equal to the
square-root of the sum of the squares of the m elements, and each element is
less-than-or-equal-to the largest element of A, then
3-93
3 Fixed-Point Topics
This becomes useful in fixed-point where the elements of A are often very
close to the maximum value attainable by the data type, so we can set a tight
upper bound without knowing the values of A. This is important because
we want to set an output type for R with a minimum number of bits, only
knowing the upper bound of the data type of A. You can use fi method
upperbound to get this value.
Attaining the maximum is easy and common. The maximum will occur when
all elements get rotated into a single element, like the following matrix with
orthogonal columns:
A = [7 -7 7 7
7 7 -7 7
7 -7 -7 -7
7 7 7 -7];
Its maximum value is 7 and its number of rows is m=4, so we expect that
the maximum value in R will be bounded by max(abs(A(:)))*sqrt(m) =
7*sqrt(4) = 14. Since A in this example is orthogonal, each column gets
rotated to the max value on the diagonal.
niter = 52;
3-94
Perform QR Factorization Using CORDIC
[Q,R] = cordicqr(A,niter)
Q =
R =
m = 9; n = 5;
A = ones(m,n)
niter = 52;
[Q,R] = cordicqr(A,niter)
A =
1 1 1 1 1
1 1 1 1 1
1 1 1 1 1
1 1 1 1 1
1 1 1 1 1
1 1 1 1 1
1 1 1 1 1
1 1 1 1 1
3-95
3 Fixed-Point Topics
1 1 1 1 1
Q =
Columns 1 through 7
Columns 8 through 9
0.0056 -0.0921
-0.5069 -0.1799
0.0359 0.3122
-0.2351 -0.0175
-0.2001 0.0610
-0.0939 -0.6294
0.7646 -0.2849
0.2300 0.2820
0 0.5485
R =
3-96
Perform QR Factorization Using CORDIC
0 0 0 0 0
A benefit of increasing the word length is that it allows for the maximum
possible precision for a given word length. A disadvantage is that the optimal
word length may not correspond to a native type on your processor (e.g.
increasing from 16 to 18 bits), or you may have to increase to the next larger
native word size which could be quite large (e.g. increasing from 16 to 32 bits,
when you only needed 18).
If you have a fixed-point input matrix A, you can define fixed-point output R
with the growth defined in the previous section.
3-97
3 Fixed-Point Topics
A = sfi(X)
A =
m = size(A,1)
m =
The growth factor is 1.6468 times the square-root of the number of rows of A.
The bit growth is the next integer above the base-2 logarithm of the growth.
bit_growth =
Initialize R with the same values as A, and a word length increased by the bit
growth.
3-98
Perform QR Factorization Using CORDIC
R =
niter = get(R,'WordLength') - 1
[Q,R] = cordicqr(R, niter)
niter =
17
Q =
R =
3-99
3 Fixed-Point Topics
double(Q)*double(Q')
ans =
err =
1.0e-03 *
Increasing Precision in R
3-100
Perform QR Factorization Using CORDIC
A = sfi(X,8,0)
A =
m = size(A,1)
m =
3-101
3 Fixed-Point Topics
The necessary growth is 1.6468 times the square-root of the number of rows
of A.
bit_growth = ceil(log2(cordic_growth_constant*sqrt(m)))
bit_growth =
Initialize R with the same values as A, and allow for bit growth like you did in
the previous section.
R =
niter = get(R,'WordLength') - 1;
[Q,R] = cordicqr(R, niter)
Q =
3-102
Perform QR Factorization Using CORDIC
R =
257 126 -1 -1
0 225 151 -148
0 0 211 104
0 0 0 -180
Notice that R is returned with integer values because you left the fraction
length of R at 0, the same as the fraction length of A.
The scaling of the least-significant bit (LSB) of A is 1, and you can see that
the error is proportional to the LSB.
err = double(Q)*double(R)-double(A)
err =
3-103
3 Fixed-Point Topics
increase the word length to 32 and set the fraction length to 22, which leaves
10 bits in the integer part.
R =
niter = get(R,'WordLength') - 1;
[Q,R] = cordicqr(R, niter)
Q =
R =
3-104
Perform QR Factorization Using CORDIC
err = double(Q)*double(R)-double(A)
err =
1.0e-05 *
The number of bits you choose for fraction length will depend on the precision
requirements for your particular algorithm.
Thus, we can make our code more usable by not requiring the number of
iterations to be input, and assuming that we want the most precision possible
by changing cordicqr to use this default for niter.
3-105
3 Fixed-Point Topics
A disadvantage of doing this is that this makes a section of our code dependent
on data type. However, an advantage is that the function is much more
convenient to use because you don’t have to specify niter if you don’t want
to, and the main algorithm is still data-type independent. Similar to picking
an optimal output type for Q, you can do this kind of input parsing in the
beginning of a function and leave the main algorithm data-type independent.
A = [7 -7 7 7
7 7 -7 7
7 -7 -7 -7
7 7 7 -7];
[Q,R] = cordicqr(A)
Q =
R =
3-106
Perform QR Factorization Using CORDIC
When you compare the results from cordicqr and the QR function in
MATLAB, you will notice that the QR factorization is not unique. It is only
important that Q is orthogonal, R is upper triangular, and Q*R - A is small.
m = 3;
A = ones(m)
A =
1 1 1
1 1 1
1 1 1
[Q0,R0] = qr(A)
Q0 =
R0 =
3-107
3 Fixed-Point Topics
0 0 0
0 0 0
[Q,R] = cordicqr(A)
Q =
R =
Notice that the elements of Q from function cordicqr are different from Q0
from built-in QR. However, both results satisfy the requirement that Q is
orthogonal:
Q0*Q0'
ans =
1.0000 0.0000 0
0.0000 1.0000 0
0 0 1.0000
Q*Q'
3-108
Perform QR Factorization Using CORDIC
ans =
Q0*R0 - A
ans =
1.0e-15 *
Q*R - A
ans =
1.0e-15 *
Given matrices A and B, you can use the QR factorization to solve for X in
the following equation:
A*X = B.
3-109
3 Fixed-Point Topics
If A has more rows than columns, then X will be the least-squares solution. If
X and B have more than one column, then several solutions can be computed
at the same time. If A = Q*R is the QR factorization of A, then the solution
can be computed by back-solving
R*X = C
You can verify the algorithm with this example. Let A be a random 3-by-3
matrix, and B be a random 3-by-2 matrix.
B = [-0.9286 0.3575
0.6983 0.5155
0.8680 0.4863];
[Q,R] = cordicqr(A)
3-110
Perform QR Factorization Using CORDIC
Q =
R =
[R,C] = cordicrc(A,B)
R =
C =
-0.3068 -0.7795
-1.1897 -0.1173
-0.7706 -0.0926
Subtract, and you will see that the error difference is on the order of roundoff.
Q'*B - C
ans =
3-111
3 Fixed-Point Topics
1.0e-15 *
-0.0555 0.3331
0 0
0.1110 0.2914
A = sfi(A)
A =
B = sfi(B)
B =
-0.9286 0.3575
0.6983 0.5155
0.8680 0.4863
The necessary growth is 1.6468 times the square-root of the number of rows
of A.
3-112
Perform QR Factorization Using CORDIC
bit_growth = ceil(log2(cordic_growth_constant*sqrt(m)))
bit_growth =
Initialize R with the same values as A, and allow for bit growth.
R =
The growth in C is the same as R, so initialize C and allow for bit growth
the same way.
C =
-0.9286 0.3575
0.6983 0.5155
0.8680 0.4863
3-113
3 Fixed-Point Topics
[R,C] = cordicrc(R,C)
R =
C =
-0.3068 -0.7796
-1.1898 -0.1175
-0.7706 -0.0926
B =
3-114
Perform QR Factorization Using CORDIC
1 0 0
0 1 0
0 0 1
[R,C] = cordicrc(A,B)
R =
C =
Then C is orthogonal
C'*C
ans =
and R = C*A
R - C*A
ans =
3-115
3 Fixed-Point Topics
1.0e-15 *
Fixed-Point Designer™
MATLAB
3-116
Perform QR Factorization Using CORDIC
3-117
3 Fixed-Point Topics
load A_3_by_3_for_cordicqr_demo.mat
niter=32;
[Q,R] = cordicqr_makeplots(A,niter)
References
3 Gene H. Golub and Charles F. Van Loan, Matrix Computations, 3rd ed,
Johns Hopkins University Press, 1996, section 5.2.3 Givens QR Methods.
6 Musheng Wei and Qiaohua Liu, "On growth factors of the modified
Gram-Schmidt algorithm," Numerical Linear Algebra with Applications,
Vol. 15, issue 7, September 2008, pp. 621-636.
Cleanup
fipref(originalFipref);
globalfimath(originalGlobalFimath);
close all
3-118
Perform QR Factorization Using CORDIC
3-119
3 Fixed-Point Topics
Introduction
3-120
Compute Square Root Using CORDIC
where
if , and otherwise.
•
•
•
where
Note also that for square root we will use only the result.
3-121
3 Fixed-Point Topics
z(:) = z - atanhLookupTable(idx);
else
x(:) = x - ytmp;
y(:) = y - xtmp;
z(:) = z + atanhLookupTable(idx);
end
if idx==k
xtmp = bitsra(x, idx); % multiply by 2^(-idx)
ytmp = bitsra(y, idx); % multiply by 2^(-idx)
if y < 0
x(:) = x + ytmp;
y(:) = y + xtmp;
z(:) = z - atanhLookupTable(idx);
else
x(:) = x - ytmp;
y(:) = y - xtmp;
z(:) = z + atanhLookupTable(idx);
end
k = 3*k + 1;
end
end % idx loop
The judicious choice of initial values allows the CORDIC kernel hyperbolic
vectoring mode algorithm to compute square root.
• is set to .
• is set to .
3-122
Compute Square Root Using CORDIC
Note: for square root, and atanhLookupTable have no impact on the result.
Hence, and atanhLookupTable are not used.
if idx==k
xtmp = bitsra(x, idx); % multiply by 2^(-idx)
ytmp = bitsra(y, idx); % multiply by 2^(-idx)
if y < 0
x(:) = x + ytmp;
y(:) = y + xtmp;
else
x(:) = x - ytmp;
3-123
3 Fixed-Point Topics
y(:) = y - xtmp;
end
k = 3*k + 1;
end
end % idx loop
Example
step = 2^-7;
v_fix = fi(0.5:step:(2-step), 1, 20); % fixed-point inputs in range [.5, 2)
niter = 10; % number of CORDIC iterations
x_sqr = cordicsqrt(v_fix, niter);
% Get the Real World Value (RWV) of the CORDIC outputs for comparison
% and plot the error between the MATLAB reference and CORDIC sqrt values
x_cdc = double(x_sqr); % CORDIC results (scaled by An_hp)
v_ref = double(v_fix); % Reference floating-point input values
x_ref = sqrt(v_ref); % MATLAB reference floating-point results
figure;
subplot(211);
plot(v_ref, x_cdc, 'r.', v_ref, x_ref, 'b-');
legend('CORDIC', 'Reference', 'Location', 'SouthEast');
title('CORDIC Square Root (In-Range) and MATLAB Reference Results');
subplot(212);
absErr = abs(x_ref - x_cdc);
plot(v_ref, absErr);
title('Absolute Error (vs. MATLAB SQRT Reference Results)');
3-124
Compute Square Root Using CORDIC
Many square root algorithms normalize the input value, , to within the range
of [0.5, 2) range. This pre-processing is typically done using a fixed word
length normalization, and can be used to support small as well as large input
value ranges.
Thus:
In the CORDICSQRT function, the values for and , described above, are
found during normalization of the input . is the number of leading zero most
significant bits (MSBs) in the binary representation of the input . These
values are found through a series of bitwise logic and shifts. Note: because
must be even, if the number of leading zero MSBs is odd, one additional bit
shift is made to make even. The resulting value after these shifts is the
value .
Example
Compute the square root of 10-bit fixed-point input data with a small
non-negative range using CORDIC. Compare the CORDIC-based algorithm
results to the floating-point MATLAB reference results over the same input
range.
step = 2^-8;
u_ref = 0:step:(0.5-step); % Input array (small range of values)
u_in_arb = fi(u_ref,0,10); % 10-bit unsigned fixed-point input data values
u_len = numel(u_ref);
3-125
3 Fixed-Point Topics
Example
Compute the square root of 16-bit fixed-point input data with a large positive
range using CORDIC. Compare the CORDIC-based algorithm results to the
floating-point MATLAB reference results over the same input range.
3-126
Compute Square Root Using CORDIC
figure;
subplot(211);
plot(u_ref, results(:,1), 'r.', u_ref, results(:,2), 'b-');
legend('CORDIC', 'Reference', 'Location', 'SouthEast');
title('CORDIC Square Root (Large Input Range) and MATLAB Reference Results'
axis([0 2500 0 55]);
subplot(212);
absErr = abs(results(:,2) - results(:,1));
plot(u_ref, absErr);
title('Absolute Error (vs. MATLAB SQRT Reference Results)');
References
3-127
3 Fixed-Point Topics
Introduction
3-128
Convert Cartesian to Polar Using CORDIC Vectoring Kernel
As approaches :
Where:
3-129
3 Fixed-Point Topics
else
x(:) = accumpos(x, ytmp);
y(:) = accumneg(y, xtmp);
z(:) = accumpos(z, inpLUT(idx));
end
xtmp = bitsra(x, idx); % bit-shift-right for multiply by 2^(-idx)
ytmp = bitsra(y, idx); % bit-shift-right for multiply by 2^(-idx)
end
The judicious choice of initial values allows the CORDIC kernel vectoring
mode algorithm to directly compute the magnitude and angle .
•
•
•
•
Example
3-130
Convert Cartesian to Polar Using CORDIC Vectoring Kernel
arithmetic unit that can perform add, subtract, shift, and memory operations.
With such a device, you could implement the CORDIC vectoring kernel to
efficiently compute magnitude and angle from the input (X,Y) coordinate
values, without the use of multiplies or large lookup tables.
% Get the Real World Value (RWV) of the CORDIC outputs for comparison
% and plot the error between the (magnitude, angle) values
AnGain = prod(sqrt(1+2.^(-2*(0:(niters-1))))); % CORDIC gain
x_c2p_RWV = (1/AnGain) .* double(x_c2p); % Magnitude (scaled by CORDIC g
z_c2p_RWV = (pi/2) .* double(z_c2p); % Angles (in radian units)
[thRWV,rRWV] = cart2pol(double(inXfix), double(inYfix)); % MATLAB reference
magnitudeErr = rRWV - x_c2p_RWV;
angleErr = thRWV - z_c2p_RWV;
figure;
subplot(411);
plot(thNorm, x_c2p_RWV);
axis([-1 1 0.25 0.5]);
title('CORDIC Magnitude (X) Values');
3-131
3 Fixed-Point Topics
subplot(412);
plot(thNorm, magnitudeErr);
title('Error between Magnitude Reference Values and X Values');
subplot(413);
plot(thNorm, z_c2p_RWV);
title('CORDIC Angle (Z) Values');
subplot(414);
plot(thNorm, angleErr);
title('Error between Angle Reference Values and Z Values');
References
3-132
Set Data Types Using Min/Max Instrumentation
In this function the variables y and z are used as both inputs and outputs.
This is an important pattern because:
• You can set the data type of y and z outside the function, thus allowing you
to re-use the function for both fixed-point and floating-point types.
• The generated C code will create y and z as references in the function
argument list. For more information about this pattern, see the
3-133
3 Fixed-Point Topics
documentation under Code Generation from MATLAB® > User’s Guide >
Generating Efficient and Reusable Code > Generating Efficient Code >
Eliminating Redundant Copies of Function Inputs.
Run the following code to copy the test function into a temporary directory so
this example doesn’t interfere with your own work.
tempdirObj = fidemo.fiTempdir('fi_instrumentation_fixed_point_filter_demo')
copyfile(fullfile(matlabroot,'toolbox','fixedpoint','fidemos','+fidemo',...
'fi_2nd_order_df2t_filter.m'),'.','f');
Run the following code to capture current states, and reset the global states.
FIPREF_STATE = get(fipref);
reset(fipref)
In this example, the requirements of the design determine the data type of
input x. These requirements are signed, 16-bit, and fractional.
N = 256;
x = fi(zeros(N,1),1,16,15);
The requirements of the design also determine the fixed-point math for a DSP
target with a 40-bit accumulator. This example uses floor rounding and wrap
overflow to produce efficient generated code.
F = fimath('RoundingMethod','Floor',...
'OverflowAction','Wrap',...
'ProductMode','KeepLSB',...
'ProductWordLength',40,...
'SumMode','KeepLSB',...
'SumWordLength',40);
[num,den] = butter(2,0.125)
3-134
Set Data Types Using Min/Max Instrumentation
The values of the coefficients influence the range of the values that will be
assigned to the filter output and states.
2. Attach fimath with floor rounding and wrap overflow settings to control
arithmetic, which leads to more efficient C code.
b = fi(num,1,16); b.fimath = F;
a = fi(den,1,16); a.fimath = F;
B = coder.Constant(b);
A = coder.Constant(a);
The values of the coefficients and values of the inputs determine the data
types of output y and state vector z. Create them with a scaled double
datatype so their values will attain full range and you can identify potential
overflows and propose data types.
yisd = fi(zeros(N,1),1,16,15,'DataType','ScaledDouble','fimath',F);
zisd = fi(zeros(2,1),1,16,15,'DataType','ScaledDouble','fimath',F);
To instrument the MATLAB® code, you create a MEX function from the
MATLAB® function using the buildInstrumentedMex command. The
inputs to buildInstrumentedMex are the same as the inputs to fiaccel,
but buildInstrumentedMex has no fi-object restrictions. The output of
3-135
3 Fixed-Point Topics
Use the '-o' option to name the MEX function that is generated. If you
do not use the '-o' option, then the MEX function is the name of the
MATLAB® function with '_mex' appended. You can also name the MEX
function the same as the MATLAB® function, but you need to remember that
MEX functions take precedence over MATLAB® functions and so changes
to the MATLAB® function will not run until either the MEX function is
re-generated, or the MEX function is deleted and cleared.
The test bench for this system is set up to run chirp and step signals. In
general, test benches for systems should cover a wide range of input signals.
The first test bench uses a chirp input. A chirp signal is a good representative
input because it covers a wide range of frequencies.
ychirp = filter_scaled_double(x,yisd,zisd);
3-136
Set Data Types Using Min/Max Instrumentation
The plot of the filtered chirp signal shows the lowpass behavior of the filter
with these particular coefficients. Low frequencies are passed through and
higher frequencies are attenuated.
clf
plot(t,x,'c',t,ychirp,'bo-')
title('Chirp')
legend('Input','Scaled-double output')
figure(gcf); drawnow;
3-137
3 Fixed-Point Topics
Potential overflows are only displayed for fi objects with Scaled Double data
type.
This particular design is for a DSP, where the word lengths are fixed, so use
the proposeFL flag to propose fraction lengths.
The next test bench is run with a step input. A step input is a good
representative input because it is often used to characterize the behavior
of a system.
xstep = [ones(N/2,1);-ones(N/2,1)];
x(:) = xstep;
The instrumentation results are accumulated until they are cleared with
clearInstrumentationResults.
ystep = filter_scaled_double(x,yisd,zisd);
clf
plot(t,x,'c',t,ystep,'bo-')
title('Step')
legend('Input','Scaled-double output')
figure(gcf); drawnow;
3-138
Set Data Types Using Min/Max Instrumentation
Even though the inputs for step and chirp inputs are both full range as
indicated by x at 100 percent current range in the instrumented code
generation report, the step input causes overflow while the chirp input did
not. This is an illustration of the necessity to have many different inputs for
your test bench. For the purposes of this example, only two inputs were used,
but real test benches should be more thorough.
At this point in the workflow, you use true fixed-point types (as opposed to
the scaled double types that were used in the earlier step of determining
data types).
yi = fi(zeros(N,1),1,16,14,'fimath',F);
zi = fi(zeros(2,1),1,16,14,'fimath',F);
3-139
3 Fixed-Point Topics
After converting to fixed-point, run the test bench again with fixed-point
inputs to validate the design.
Run the fixed-point algorithm with a chirp input to validate the design.
x(:) = xchirp;
[y,z] = filter_fixed_point(x,yi,zi);
[ysd,zsd] = filter_scaled_double(x,yisd,zisd);
err = double(y) - double(ysd);
clf
subplot(211);plot(t,x,'c',t,ysd,'bo-',t,y,'mx')
xlabel('Time (s)');
ylabel('Amplitude')
legend('Input','Scaled-double output','Fixed-point output');
title('Fixed-Point Chirp')
subplot(212);plot(t,err,'r');title('Error');xlabel('t'); ylabel('err');
figure(gcf); drawnow;
Inspect the variables and intermediate results to ensure that the min/max
values are within range.
showInstrumentationResults filter_fixed_point
Run the fixed-point algorithm with a step input to validate the design.
Run the following code to clear the previous instrumentation results to see
only the effects of running the step input.
3-140
Set Data Types Using Min/Max Instrumentation
clearInstrumentationResults filter_fixed_point
Run the step input through the fixed-point filter and compare with the output
of the scaled double filter.
x(:) = xstep;
[y,z] = filter_fixed_point(x,yi,zi);
[ysd,zsd] = filter_scaled_double(x,yisd,zisd);
err = double(y) - double(ysd);
Plot the fixed-point outputs against the scaled-double outputs to verify that
they meet your design criteria.
clf
subplot(211);plot(t,x,'c',t,ysd,'bo-',t,y,'mx')
title('Fixed-Point Step');
legend('Input','Scaled-double output','Fixed-point output')
subplot(212);plot(t,err,'r');title('Error');xlabel('t'); ylabel('err');
figure(gcf); drawnow;
Inspect the variables and intermediate results to ensure that the min/max
values are within range.
showInstrumentationResults filter_fixed_point
fipref(FIPREF_STATE);
clearInstrumentationResults filter_fixed_point
clearInstrumentationResults filter_scaled_double
clear fi_2nd_order_df2t_filter_fixed_instrumented
clear fi_2nd_order_df2t_filter_float_instrumented
tempdirObj.cleanUp;
3-141
3 Fixed-Point Topics
3-142
Convert Fast Fourier Transform (FFT) to Fixed Point
Run the following code to copy functions from the Fixed-Point Designer™
examples directory into a temporary directory so this example doesn’t
interfere with your own work.
tempdirObj = fidemo.fiTempdir('fi_radix2fft_demo');
copyfile(fullfile(matlabroot,'toolbox','fixedpoint','fidemos','+fidemo',...
'fi_m_radix2fft_algorithm1_6_2.m'),'.','f');
copyfile(fullfile(matlabroot,'toolbox','fixedpoint','fidemos',...
'fi_m_radix2fft_withscaling.m'),'.','f');
Run the following code to capture current states, and reset the global states.
FIPREF_STATE = get(fipref);
reset(fipref)
figure(gcf); clf
subplot(211); plot(t,real(x0),'b.-'); xlabel('Time (s)'); ylabel('Amplitude
3-143
3 Fixed-Point Topics
The peaks at 0.2 and 0.5 Hz in the frequency plot correspond to the two
sinusoids of the time-domain signal at those frequencies.
Note the reflected peaks at 3.5 and 3.8 Hz. When the input to an FFT is
real-valued, as it is in this case, then the output y is conjugate-symmetric:
There are many different implementations of the FFT, each having its own
costs and benefits. You may find that a different algorithm is better for your
application than the one given here. This algorithm is used to provide you
with an example of how you might begin your own exploration.
function w = fi_radix2twiddles(n)
t = log2(n);
if floor(t) ~= t
error('N must be an exact power of two.');
end
w = zeros(n-1,1);
k=1;
3-144
Convert Fast Fourier Transform (FFT) to Fixed Point
L=2;
% Equation 1.4.11, p. 34
while L<=n
theta = 2*pi/L;
% Algorithm 1.4.1, p. 23
for j=0:(L/2 - 1)
w(k) = complex( cos(j*theta), -sin(j*theta) );
k = k + 1;
end
L = L*2;
end
figure(gcf);clf
w0 = fidemo.fi_radix2twiddles(n);
polar(angle(w0),abs(w0),'o')
title('Twiddle Factors: Complex roots of unity')
function x = fi_m_radix2fft_algorithm1_6_2(x, w)
n = length(x); t = log2(n);
x = fidemo.fi_bitreverse(x,n);
for q=1:t
L = 2^q; r = n/L; L2 = L/2;
for k=0:(r-1)
for j=0:(L2-1)
temp = w(L2-1+j+1) * x(k*L+j+L2+1);
x(k*L+j+L2+1) = x(k*L+j+1) - temp;
x(k*L+j+1) = x(k*L+j+1) + temp;
end
end
end
3-145
3 Fixed-Point Topics
y = fi_m_radix2fft_algorithm1_6_2(x0, w0);
Because the error is within tolerance of the MATLAB built-in FFT function,
you know you have correctly implemented the algorithm.
Now, try converting the data to fixed-point and see if the algorithm still looks
good. In this first pass, you use all the defaults for signed fixed-point data
by using the sfi constructor.
Note that the magnitude plot (center) of the fixed-point FFT does not resemble
the plot of the built-in FFT. The error (bottom plot) is much larger than what
you would expect to see for round off error, so it is likely that overflow has
occurred.
To instrument the MATLAB® code, you create a MEX function from the
MATLAB® function using the buildInstrumentedMex command. The
inputs to buildInstrumentedMex are the same as the inputs to fiaccel,
3-146
Convert Fast Fourier Transform (FFT) to Fixed Point
The '-o' option is used to name the MEX function that is generated. If
the '-o' option is not used, then the MEX function is the name of the
MATLAB® function with '_mex' appended. You can also name the MEX
function the same as the MATLAB® function, but you need to remember that
MEX functions take precedence over MATLAB® functions and so changes
to the MATLAB® function will not run until either the MEX function is
re-generated, or the MEX function is deleted and cleared.
Create the input with a scaled double datatype so its values will attain full
range and you can identify potential overflows.
x_scaled_double = fi(x0,'DataType','ScaledDouble');
buildInstrumentedMex fi_m_radix2fft_algorithm1_6_2 ...
-o fft_instrumented -args {x_scaled_double w}
y_scaled_double = fft_instrumented(x_scaled_double,w);
showInstrumentationResults fft_instrumented
You can see from the instrumentation results that there were overflows when
assigning into the variable x.
3-147
3 Fixed-Point Topics
When you scale only the input to the first stage of a length-n FFT by 1/n,
you obtain a noise-to-signal ratio proportional to n^2 [Oppenheim & Schafer
1989, equation 9.101], [Welch 1969].
However, if you scale the input to each of the stages of the FFT by 1/2,
you can obtain an overall scaling of 1/n and produce a noise-to-signal ratio
proportional to n [Oppenheim & Schafer 1989, equation 9.105], [Welch 1969].
function x = fi_m_radix2fft_withscaling(x, w)
n = length(x); t = log2(n);
x = fidemo.fi_bitreverse(x,n);
% Generate index variables as integer constants so they are not computed in
% the loop.
LL = int32(2.^(1:t)); rr = int32(n./LL); LL2 = int32(LL./2);
for q=1:t
L = LL(q); r = rr(q); L2 = LL2(q);
for k=0:(r-1)
for j=0:(L2-1)
temp = w(L2-1+j+1) * x(k*L+j+L2+1);
x(k*L+j+L2+1) = bitsra(x(k*L+j+1) - temp, 1);
x(k*L+j+1) = bitsra(x(k*L+j+1) + temp, 1);
end
end
end
x = sfi(x0);
w = sfi(w0);
y = fi_m_radix2fft_withscaling(x,w);
3-148
Convert Fast Fourier Transform (FFT) to Fixed Point
You can see that the scaled fixed-point FFT algorithm now matches the
built-in FFT to a tolerance that is expected for 16-bit fixed-point data.
References
fipref(FIPREF_STATE);
clearInstrumentationResults fft_instrumented
clear fft_instrumented
tempdirObj.cleanUp;
3-149
3 Fixed-Point Topics
The example focuses on detecting large scale limit cycles due to overflow
with zero inputs and highlights the conditions that are sufficient to prevent
such oscillations.
References:
We observe that the system is stable by observing that the eigenvalues of the
state-transition matrix A have magnitudes less than 1.
ans =
0.5000 + 0.5000i
0.5000 - 0.5000i
Filter Implementation
type(fullfile(matlabroot,'toolbox','fixedpoint','fidemos','+fidemo','fisiso
3-150
Detect Limit Cycles in Fixed-Point State-Space Systems
y = x;
z(:,2:length(x)+1) = 0;
for k=1:length(x)
y(k) = C*z(:,k) + D*x(k);
z(:,k+1) = A*z(:,k) + B*x(k);
end
Floating-Point Filter
First, we choose random states within the unit square and observe where
they are projected after one step of being multiplied by the state-transition
matrix A.
rng('default');
clf
x1 = [-1 1 1 -1 -1];
y1 = [-1 -1 1 1 -1];
plot(x1,y1,'c')
axis([-1.5 1.5 -1.5 1.5]); axis square; grid;
hold on
r = 2*rand(2,1000)-1;
pr = A*r;
plot(pr(1,:),pr(2,:),'.')
3-151
3 Fixed-Point Topics
Drive the filter with a random initial state, normalized to be inside the unit
square, with the input all zero, and run the filter.
Note that some of the states wander outside the unit square, and that they
eventually wind down to the zero state at the origin, z=[0;0].
x = zeros(10,1);
zi = [0;0];
q = quantizer([16 15]);
for k=1:20
y = x;
zi(:) = randquant(q,size(A,1),1);
[y,zf] = fidemo.fisisostatespacefilter(A,B,C,D,x,zi);
plot(zf(1,:), zf(2,:),'go-','markersize',8);
end
title('Double-Precision State Sequence Plot');
xlabel('z1'); ylabel('z2')
State Trajectory
Because the eigenvalues are less than one in magnitude, the system is stable,
and all initial states wind down to the origin with zero input. However, the
eigenvalues don’t tell the whole story about the trajectory of the states, as
in this example, where the states were projected outward first, before they
start to contract.
svd(A)
ans =
3-152
Detect Limit Cycles in Fixed-Point State-Space Systems
1.4604
0.3424
The MATLAB® code for the filter remains the same. It becomes a fixed-point
filter because we drive it with fixed-point inputs.
For the sake of illustrating overflow oscillation, we are choosing product and
sum data types that will overflow.
rng('default');
F = fimath('OverflowAction','Wrap',...
'ProductMode','SpecifyPrecision',...
'ProductWordLength',16,'ProductFractionLength',15,...
'SumMode','SpecifyPrecision',...
'SumWordLength',16,'SumFractionLength',15);
A = fi(A,'fimath',F)
B = fi(B,'fimath',F)
C = fi(C,'fimath',F)
D = fi(D,'fimath',F)
A =
0 1.0000
-0.5000 1.0000
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: SpecifyPrecision
3-153
3 Fixed-Point Topics
ProductWordLength: 16
ProductFractionLength: 15
SumMode: SpecifyPrecision
SumWordLength: 16
SumFractionLength: 15
CastBeforeSum: true
B =
0
1
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: SpecifyPrecision
ProductWordLength: 16
ProductFractionLength: 15
SumMode: SpecifyPrecision
SumWordLength: 16
SumFractionLength: 15
CastBeforeSum: true
C =
1 0
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: SpecifyPrecision
ProductWordLength: 16
3-154
Detect Limit Cycles in Fixed-Point State-Space Systems
ProductFractionLength: 15
SumMode: SpecifyPrecision
SumWordLength: 16
SumFractionLength: 15
CastBeforeSum: true
D =
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: SpecifyPrecision
ProductWordLength: 16
ProductFractionLength: 15
SumMode: SpecifyPrecision
SumWordLength: 16
SumFractionLength: 15
CastBeforeSum: true
Again, we choose random states within the unit square and observe where
they are projected after one step of being multiplied by the state-transition
matrix A. The difference is that this time matrix A is fixed-point.
Note that the triangles that projected out of the square before in floating-point,
are now wrapped back into the interior of the square.
clf
r = 2*rand(2,1000)-1;
pr = A*r;
plot([-1 1 1 -1 -1],[-1 -1 1 1 -1],'c')
axis([-1.5 1.5 -1.5 1.5]); axis square; grid;
hold on
3-155
3 Fixed-Point Topics
plot(pr(1,:),pr(2,:),'.')
The only difference between this and the previous code is that we are driving
it with fixed-point data types.
x = fi(zeros(10,1),1,16,15,'fimath',F);
zi = fi([0;0],1,16,15,'fimath',F);
q = assignmentquantizer(zi);
e = double(eps(zi));
rng('default');
for k=1:20
y = x;
zi(:) = randquant(q,size(A,1),1);
[y,zf] = fidemo.fisisostatespacefilter(A,B,C,D,x,zi);
if abs(double(zf(end)))>0.5, c='ro-'; else, c='go-'; end
plot(zf(1,:), zf(2,:),c,'markersize',8);
end
title('Fixed-Point State Sequence Plot');
xlabel('z1'); ylabel('z2')
Trying this for other randomly chosen initial states illustrates that once a
state enters one of the triangular regions, then it is projected into the other
triangular region, and back and forth, and never escapes.
There are two sufficient conditions to prevent overflow limit cycles in a system:
Note that for the current representation, the second condition does not hold.
3-156
Detect Limit Cycles in Fixed-Point State-Space Systems
Now the projection of random initial states inside the unit square all contract
uniformly. This is the result of the state transition matrix A2 being normal.
The states are also rotated by 90 degrees counterclockwise.
clf
r = 2*rand(2,1000)-1;
pr = A2*r;
plot([-1 1 1 -1 -1],[-1 -1 1 1 -1],'c')
axis([-1.5 1.5 -1.5 1.5]); axis square; grid;
hold on
plot(pr(1,:),pr(2,:),'.')
Plotting the state sequences again for the same initial states as before we see
that the outputs now spiral towards the origin.
x = fi(zeros(10,1),1,16,15,'fimath',F);
zi = fi([0;0],1,16,15,'fimath',F);
q = assignmentquantizer(zi);
e = double(eps(zi));
rng('default');
for k=1:20
y = x;
3-157
3 Fixed-Point Topics
zi(:) = randquant(q,size(A,1),1);
[y,zf] = fidemo.fisisostatespacefilter(A2,B2,C2,D2,x,zi);
if abs(double(zf(end)))>0.5, c='ro-'; else, c='go-'; end
plot(zf(1,:), zf(2,:),c,'markersize',8);
end
title('Normal-Form Fixed-Point State Sequence Plot');
xlabel('z1'); ylabel('z2')
Trying this for other randomly chosen initial states illustrates that there is no
region from which the filter is unable to recover.
3-158
Compute Quantization Error
First, a random signal is created that spans the range of the quantizer.
Next, the signal is quantized, respectively, with rounding methods ’fix’, ’floor’,
’ceil’, ’nearest’, and ’convergent’, and the statistics of the signal are estimated.
First we create a uniformly distributed random signal that spans the domain
-1 to 1 of the fixed-point quantizers that we will look at.
q = quantizer([8 7]);
r = realmax(q);
u = r*(2*rand(50000,1) - 1); % Uniformly distributed (-1,1)
xi=linspace(-2*eps(q),2*eps(q),256);
Notice that with ’fix’ rounding, the probability density function is twice as wide
as the others. For this reason, the variance is four times that of the others.
q = quantizer('fix',[8 7]);
err = quantize(q,u) - u;
f_t = errpdf(q,xi);
mu_t = errmean(q);
v_t = errvar(q);
% Theoretical variance = eps(q)^2 / 3
% Theoretical mean = 0
fidemo.qerrordemoplot(q,f_t,xi,mu_t,v_t,err)
3-159
3 Fixed-Point Topics
Floor rounding is often called truncation when used with integers and
fixed-point numbers that are represented in two’s complement. It is the most
common rounding mode of DSP processors because it requires no hardware to
implement. Floor does not produce quantized values that are as close to the
true values as ROUND will, but it has the same variance, and small signals
that vary in sign will be detected, whereas in ROUND they will be lost.
q = quantizer('floor',[8 7]);
err = quantize(q,u) - u;
f_t = errpdf(q,xi);
mu_t = errmean(q);
v_t = errvar(q);
% Theoretical variance = eps(q)^2 / 12
% Theoretical mean = -eps(q)/2
fidemo.qerrordemoplot(q,f_t,xi,mu_t,v_t,err)
q = quantizer('ceil',[8 7]);
err = quantize(q,u) - u;
f_t = errpdf(q,xi);
mu_t = errmean(q);
v_t = errvar(q);
% Theoretical variance = eps(q)^2 / 12
% Theoretical mean = eps(q)/2
fidemo.qerrordemoplot(q,f_t,xi,mu_t,v_t,err)
3-160
Compute Quantization Error
Round is more accurate than floor, but all values smaller than eps(q) get
rounded to zero and so are lost.
q = quantizer('nearest',[8 7]);
err = quantize(q,u) - u;
f_t = errpdf(q,xi);
mu_t = errmean(q);
v_t = errvar(q);
% Theoretical variance = eps(q)^2 / 12
% Theoretical mean = 0
fidemo.qerrordemoplot(q,f_t,xi,mu_t,v_t,err)
q = quantizer('convergent',[8 7]);
err = quantize(q,u) - u;
f_t = errpdf(q,xi);
mu_t = errmean(q);
v_t = errvar(q);
% Theoretical variance = eps(q)^2 / 12
% Theoretical mean = 0
3-161
3 Fixed-Point Topics
fidemo.qerrordemoplot(q,f_t,xi,mu_t,v_t,err)
f(err) = 1/eps(q), for -eps(q)/2 <= err <= eps(q)/2, and 0 otherwise
f(err) = 1/eps(q), for -eps(q)/2 < err <= eps(q)/2, and 0 otherwise
Note that the error p.d.f. of convergent is symmetric, while round is slightly
biased towards the positive.
x=[-3.5:3.5]';
[x convergent(x) nearest(x)]
ans =
3-162
Compute Quantization Error
The helper function that was used to generate the plots in this example is
listed below.
type(fullfile(matlabroot,'toolbox','fixedpoint','fidemos','+fidemo','qerror
function qerrordemoplot(q,f_t,xi,mu_t,v_t,err)
%QERRORDEMOPLOT Plot function for QERRORDEMO.
% QERRORDEMOPLOT(Q,F_T,XI,MU_T,V_T,ERR) produces the plot and display us
% the example function QERRORDEMO, where Q is the quantizer whos attribu
% being analyzed; F_T is the theoretical quantization error probability
% density function for quantizer Q computed by ERRPDF; XI is the domain
% values being evaluated by ERRPDF; MU_T is the theoretical quantization
% error mean of quantizer Q computed by ERRMEAN; V_T is the theoretical
% quantization error variance of quantizer Q computed by ERRVAR; and ERR
% is the error generated by quantizing a random signal by quantizer Q.
%
% See QERRORDEMO for examples of use.
v=10*log10(var(err));
disp(['Estimated error variance (dB) = ',num2str(v)]);
disp(['Theoretical error variance (dB) = ',num2str(10*log10(v_t))]);
disp(['Estimated mean = ',num2str(mean(err))]);
disp(['Theoretical mean = ',num2str(mu_t)]);
[n,c]=hist(err);
figure(gcf)
bar(c,n/(length(err)*(c(2)-c(1))),'hist');
line(xi,f_t,'linewidth',2,'color','r');
% Set the ylim uniformly on all plots
set(gca,'ylim',[0 max(errpdf(quantizer(q.format,'nearest'),xi)*1.1)])
legend('Estimated','Theoretical')
xlabel('err'); ylabel('errpdf')
3-163
3 Fixed-Point Topics
Setup
To assure that this example does not change your preferences or settings, this
code stores the original state, and you will restore it at the end.
This algorithm normalizes unsigned data with 8-bit bytes. Given input u > 0,
the output x is normalized such that
u = x * 2^n
where 1 <= x < 2 and n is an integer. Note that n may be positive, negative,
or zero.
3-164
Normalize Data for Lookup Tables
3-165
3 Fixed-Point Topics
And so forth, until the last element of NLZLUT is 0 and corresponds to u=255.
There are 0 leading zero bits in the 8-bit value u=11111111_2.
3-166
Normalize Data for Lookup Tables
1 1 1 1 1 1 1 1 ...
1 1 1 1 1 1 1 1 ...
1 1 1 1 1 1 1 1 ...
1 1 1 1 1 1 1 1 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0 ...
0 0 0 0 0 0 0 0]);
end
Example
3-167
3 Fixed-Point Topics
The high byte is 0 = 00000000_2. Add 1 to make an index out of it: index
= 0 + 1 = 1. The number-of-leading-zeros lookup table at index 1 indicates
that there are 8 leading zeros: NLZLUT(1) = 8. Left shift by this many bits.
Iterate once more to remove the leading zeros from the next byte.
The high byte is 77 = 01001101_2. Add 1 to make an index out of it: index =
77 + 1 = 78. The number-of-leading-zeros lookup table at index 78 indicates
that there is 1 leading zero: NLZLUT(78) = 1. Left shift by this many bits.
x = 1.001101000000000_2 = 1.203125
The value for n is the word-length of u, minus the fraction length of u, minus
the number of left shifts, minus 1.
n = 16 - 8 - 9 - 1 = -2.
[x,n] = fi_normalize_unsigned_8_bit_byte(u)
x =
1.203125
n =
-2
3-168
Normalize Data for Lookup Tables
Comparing binary values, you can see that x has the same bits as u,
left-shifted by 9 bits.
binary_representation_of_u = bin(u)
binary_representation_of_x = bin(x)
binary_representation_of_u =
0000000001001101
binary_representation_of_x =
1001101000000000
Cleanup
3-169
3 Fixed-Point Topics
Setup
To assure that this example does not change your preferences or settings, this
code stores the original state, and you will restore it at the end.
Log2 Implementation
3 Extract the upper B-bits of x. Let x_B denote the upper B-bits of x.
6 Finally, compute the output using the lookup table and linear interpolation:
3-170
Implement Fixed-Point Log2 Using Lookup Table
Function log2_lookup_table loads the lookup table of log2 values. You can
create the table by running:
B = 8;
log2_table = log2((2^(B-1) : 2^(B)) / 2^(B - 1))
3-171
3 Fixed-Point Topics
3-172
Implement Fixed-Point Log2 Using Lookup Table
Example
u = fi(linspace(0.001,20,100));
y = fi_log2lookup_8_bit_byte(u);
y_expected = log2(double(u));
%%3
clf
subplot(211)
plot(u,y,u,y_expected)
legend('Output','Expected output','Location','Best')
subplot(212)
plot(u,double(y)-y_expected,'r')
legend('Error')
figure(gcf)
Cleanup
3-173
3 Fixed-Point Topics
fipref(originalFiprefState);
3-174
Implement Fixed-Point Square Root Using Lookup Table
Setup
To assure that this example does not change your preferences or settings, this
code stores the original state, and you will restore it at the end.
3 Extract the upper B-bits of x. Let x_B denote the upper B-bits of x.
6 Finally, compute the output using the lookup table and linear interpolation:
3-175
3 Fixed-Point Topics
3-176
Implement Fixed-Point Square Root Using Lookup Table
3-177
3 Fixed-Point Topics
Example
u = fi(linspace(0,128,1000),0,16,12);
y = fi_sqrtlookup_8_bit_byte(u);
y_expected = sqrt(double(u));
3-178
Implement Fixed-Point Square Root Using Lookup Table
clf
subplot(211)
plot(u,y,u,y_expected)
legend('Output','Expected output','Location','Best')
subplot(212)
plot(u,double(y)-y_expected,'r')
legend('Error')
figure(gcf)
Cleanup
3-179
3 Fixed-Point Topics
You can generate C code from the examples if you have MATLAB Coder™
software.
You can insulate your fixed-point operations from global and local fimath
settings by using the setfimath and removefimath functions. You can also
return from functions with no fimath attached to output variables. This gives
you local control over fixed-point math settings without interfering with the
settings in other functions.
MATLAB Code
function y = user_written_sum(u)
% Setup
F = fimath('RoundingMethod','Floor',...
'OverflowAction','Wrap',...
'SumMode','KeepLSB',...
'SumWordLength',32);
u = setfimath(u,F);
y = fi(0,true,32,get(u,'FractionLength'),F);
% Algorithm
for i=1:length(u)
y(:) = y + u(i);
end
% Cleanup
y = removefimath(y);
end
3-180
Set Fixed-Point Math Attributes
When you run the code, the fimath controls the arithmetic inside the
function, but the return value has no attached fimath. This is due to the use
of setfimath and removefimath inside the function user_written_sum.
>> u = fi(1:10,true,16,11);
>> y = user_written_sum(u)
y =
55
DataTypeMode: Fixed-point: binary point scaling
Signedness: Signed
WordLength: 32
FractionLength: 11
Generated C Code
If you have MATLAB Coder software, you can generate C code using the
following commands.
>> u = fi(1:10,true,16,11);
>> codegen user_written_sum -args {u} -config:lib -launchreport
Mismatched FIMATH
3-181
3 Fixed-Point Topics
>> A = fi(pi,'ProductMode','KeepLSB');
>> B = fi(2,'ProductMode','SpecifyPrecision');
>> C = A * B
To avoid this error, you can remove fimath from one of the variables in the
expression. In this example, the fimath is removed from B in the context of
the expression without modifying B itself, and the product is computed using
the fimath attached to A.
>> C = A * removefimath(B)
C =
6.283203125
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: KeepLSB
ProductWordLength: 32
SumMode: FullPrecision
If you have variables with no attached fimath, but you want to control a
particular operation, then you can attach a fimath in the context of the
expression without modifying the variables.
>> F = fimath('ProductMode','KeepLSB','OverflowAction','Wrap','RoundingMeth
3-182
Set Fixed-Point Math Attributes
>> A = fi(pi);
>> B = fi(2);
>> C = A * setfimath(B,F)
C =
6.2832
RoundingMethod: Floor
OverflowAction: Wrap
ProductMode: KeepLSB
ProductWordLength: 32
SumMode: FullPrecision
MaxSumWordLength: 128
>> B
B =
You can compute products and sums to match the accumulator of a DSP with
floor rounding and wrap overflow, and use nearest rounding and saturate
overflow on the output. To avoid mismatched fimath errors, you can remove
the fimath on the output variable when it is used in a computation with the
other variables.
3-183
3 Fixed-Point Topics
MATLAB Code
In this example, the products are 32-bits, and the accumulator is 40-bits,
keeping the least-significant-bits with floor rounding and wrap overflow like
C’s native integer rules. The output uses nearest rounding and saturate
overflow.
3-184
Set Fixed-Point Math Attributes
If you have MATLAB Coder software, you can generate C code with the
specificed hardware characteristics using the following commands.
N = 256;
t = 1:N;
xstep = [ones(N/2,1);-ones(N/2,1)];
num = [0.0299545822080925 0.0599091644161849 0.0299545822080925];
den = [1 -1.4542435862515900 0.5740619150839550];
b = fi(num,true,16);
a = fi(den,true,16);
x = fi(xstep,true,16,15);
zi = fi(zeros(2,1),true,16,14);
B = coder.Constant(b);
A = coder.Constant(a);
config_obj = coder.config('lib');
config_obj.GenerateReport = true;
config_obj.LaunchReport = true;
config_obj.TargetLang = 'C';
config_obj.GenerateComments = true;
config_obj.GenCodeOnly = true;
config_obj.HardwareImplementation.ProdBitPerChar=8;
config_obj.HardwareImplementation.ProdBitPerShort=16;
config_obj.HardwareImplementation.ProdBitPerInt=32;
config_obj.HardwareImplementation.ProdBitPerLong=40;
Generated C Code
3-185
3 Fixed-Point Topics
int40_T i0;
int16_T b_y;
/* Setup */
/* Set fimaths that are local to this function */
/* Create y with nearest rounding */
/* Algorithm */
for (j = 0; j < 256; j++) {
/* Nearest assignment into y */
i0 = 15705 * x[j] + ((int40_T)z[0] << 20);
b_y = (int16_T)((int32_T)(i0 >> 20) + ((i0 & 524288L) != 0L));
Polymorphic Code
You can write MATLAB code that can be used for both floating-point and
fixed-point types using setfimath and removefimath.
function y = user_written_function(u)
% Setup
F = fimath('RoundingMethod','Floor',...
'OverflowAction','Wrap',...
'SumMode','KeepLSB');
u = setfimath(u,F);
% Algorithm
y = u + u;
% Cleanup
y = removefimath(y);
end
3-186
Set Fixed-Point Math Attributes
When the function is called with fixed-point inputs, then fimath F is used for
the arithmetic, and the output has no attached fimath.
>> u = fi(pi/8,true,16,15,'RoundingMethod','Convergent');
>> y = user_written_function(u)
y =
0.785400390625
If you have MATLAB Coder software, you can generate C code using the
following commands.
>> u = fi(pi/8,true,16,15,'RoundingMethod','Convergent');
>> codegen user_written_function -args {u} -config:lib -launchreport
int32_T user_written_function(int16_T u)
{
/* Setup */
/* Algorithm */
/* Cleanup */
return u + u;
}
Double Inputs
function y = user_written_function(u)
3-187
3 Fixed-Point Topics
% Setup
F = fimath('RoundingMethod','Floor',...
'OverflowAction','Wrap',...
'SumMode','KeepLSB');
u = setfimath(u,F);
% Algorithm
y = u + u;
% Cleanup
y = removefimath(y);
end
When compiled with floating-point input, you get the following generated
C code.
real_T user_written_function(real_T u)
{
return u + u;
}
This function is written so that the output is created to be the same type as
the input, so both floating-point and fixed-point can be used with it.
function y = user_written_sum_polymorphic(u)
% Setup
F = fimath('RoundingMethod','Floor',...
'OverflowAction','Wrap',...
'SumMode','KeepLSB',...
'SumWordLength',32);
u = setfimath(u,F);
3-188
Set Fixed-Point Math Attributes
if isfi(u)
y = fi(0,true,32,get(u,'FractionLength'),F);
else
y = zeros(1,1,class(u));
end
% Algorithm
for i=1:length(u)
y(:) = y + u(i);
end
% Cleanup
y = removefimath(y);
end
If you have MATLAB Coder software, you can generate fixed-point C code
using the following commands.
>> u = fi(1:10,true,16,11);
>> codegen user_written_sum_polymorphic -args {u} -config:lib -launchreport
/* Setup */
y = 0;
/* Algorithm */
for (i = 0; i < 10; i++) {
y += u[i];
}
3-189
3 Fixed-Point Topics
/* Cleanup */
return y;
}
If you have MATLAB Coder software, you can generate floating-point C code
using the following commands.
>> u = 1:10;
>> codegen user_written_sum_polymorphic -args {u} -config:lib -launchreport
/* Setup */
y = 0.0;
/* Algorithm */
for (i = 0; i < 10; i++) {
y += u[i];
}
/* Cleanup */
return y;
}
>> u = int8(5);
>> codegen user_written_u_plus_u -args {u} -config:lib -launchreport
function y = user_written_u_plus_u(u)
3-190
Set Fixed-Point Math Attributes
% Setup
F = fimath('RoundingMethod','Floor',...
'OverflowAction','Wrap',...
'SumMode','KeepLSB',...
'SumWordLength',32);
u = setfimath(u,F);
% Algorithm
y = u + u;
% Cleanup
y = removefimath(y);
end
int32_T user_written_u_plus_u(int8_T u)
{
/* Setup */
/* Algorithm */
/* Cleanup */
return u + u;
}
3-191
3 Fixed-Point Topics
3-192
4
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
• You can use the fimath constructor function to create new fimath objects.
• You can use the fimath constructor function to copy an existing fimath
object.
F = fimath
F =
RoundingMethod: Nearest
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
4-2
fimath Object Construction
F = fimath;
G = F;
isequal(F,G)
ans =
4-3
4 Working with fimath Objects
The syntax
F = fimath(...'PropertyName',PropertyValue...)
allows you to set properties for a fimath object at object creation with
property name/property value pairs. Refer to “Setting fimath Properties at
Object Creation” on page 4-13.
For example, to create a fimath object that uses convergent rounding and
wraps on overflow, perform the following steps:
1 On the Home tab, in the File section, click New > Script to open the
MATLAB Editor
2 On the Editor tab, in the Edit section, click in the Insert button
group. Click the Insert fimath... to open the Insert fimath Constructor
dialog box.
3 Use the edit boxes and drop-down menus to specify the following properties
of the fimath object:
• Rounding method = Floor
• Overflow action = Wrap
• Product mode = FullPrecision
• Sum mode = FullPrecision
4-4
fimath Object Construction
4 To insert the fimath object constructor string in your file, place your
cursor at the desired location in the file. Then click OK on the Insert
fimath Constructor dialog box. Clicking OK closes the Insert fimath
Constructor dialog box and automatically populates the fimath object
constructor string in your file:
4-5
4 Working with fimath Objects
4-6
fimath Object Properties
Fraction length,
ProductFractionLength • 30 (default)
in bits, of the
• Any positive or negative integer
product data
type
Note The ProductFractionLength is the negative of the
ProductFixedExponent. Changing one property changes
the other.
4-7
4 Working with fimath Objects
Note
4-8
fimath Object Properties
4-9
4 Working with fimath Objects
Note
Note
4-10
fimath Object Properties
For details about these properties, refer to the “fi Object Properties” on page
2-19. To learn how to specify properties for fimath objects in Fixed-Point
Designer software, refer to “Setting fimath Object Properties” on page 4-13.
Ws integer length Fs
where
Fs max( Fa , Fb )
When the SumMode is set to KeepLSB, the resulting word length and fraction
length is determined by
When the SumMode is set to KeepMSB, the resulting word length and fraction
length is determined by
4-11
4 Working with fimath Objects
where
When the SumMode is set to SpecifyPrecision, you specify both the word and
fraction length or slope and bias of the sum data type with the SumWordLength
and SumFractionLength, or SumSlope and SumBias properties respectively.
W p Wa Wb
Fp Fa Fb
When the ProductMode is KeepLSB the word length and fraction length are
determined by
When the ProductMode is KeepMSB the word length and fraction length are
4-12
fimath Object Properties
For more information about how certain functions use the fimath properties,
see
For example, to set the overflow action to Saturate and the rounding method
to Convergent,
F = fimath('OverflowAction','Saturate','RoundingMethod','Convergent')
F =
RoundingMethod: Convergent
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: FullPrecision
In addition to creating a fimath object at the command line, you can also
set fimath properties using the Insert fimath Constructor dialog box.
For an example of this approach, see “Building fimath Object Constructors
in a GUI” on page 4-4.
4-13
4 Working with fimath Objects
F.RoundingMethod
ans =
Convergent
F.OverflowAction = 'Wrap'
F =
RoundingMethod: Convergent
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
4-14
fimath Properties Usage for Fixed-Point Arithmetic
The following sections discuss how fi objects with local fimath objects
interact with fi objects without local fimath.
Binary Operations
In binary fixed-point operations such as c = a + b, the following rules apply:
• If both a and b have no local fimath, the operation uses default fimath
values to perform the fixed-point arithmetic. The output fi object c also
has no local fimath.
• If either a or b has a local fimath object, the operation uses that fimath
object to perform the fixed-point arithmetic. The output fi object c has the
same local fimath object as the input.
Unary Operations
In unary fixed-point operations such as b = abs(a), the following rules apply:
• If a has no local fimath, the operation uses default fimath values to perform
the fixed-point arithmetic. The output fi object b has no local fimath.
4-15
4 Working with fimath Objects
• If a has a local fimath object, the operation uses that fimath object to
perform the fixed-point arithmetic. The output fi object b has the same
local fimath object as the input a.
When you specify a fimath object in the function call of a unary fixed-point
operation, the operation uses the fimath object you specify to perform the
fixed-point arithmetic. For example, when you use a syntax such as b =
abs(a,F) or b = sqrt(a,F), the abs and sqrt operations use the fimath
object F to compute intermediate quantities. The output fi object b always
has no local fimath.
Concatenation Operations
In fixed-point concatenation operations such as c = [a b], c = [a;b] and
c = bitconcat(a,b), the following rule applies:
For example, consider the following scenarios for the operation d = [a b c]:
4-16
fimath Properties Usage for Fixed-Point Arithmetic
All input signals to the MATLAB Function block that you treat as fi objects
associate with whatever you specify for the MATLAB Function block
fimath parameter. When you set this parameter to Same as MATLAB, your
fi objects do not have local fimath. When you set the MATLAB Function
block fimath parameter to Specify other, you can define your own set
of fimath properties for all fi objects in the MATLAB Function block to
associate with. You can choose to treat only fixed-point input signals as fi
objects or both fixed-point and integer input signals as fi objects. See “Using
fimath Objects in MATLAB Function Blocks” on page 9-71.
Binary-Point Arithmetic
The fimath object encapsulates the math properties of Fixed-Point Designer
software.
fi objects only have a local fimath object when you explicitly specify fimath
properties in the fi constructor. When you use the sfi or ufi constructor or
do not specify any fimath properties in the fi constructor, the resulting fi
object does not have any local fimath and uses default fimath values.
a = fi(pi)
a =
3.1416
a.fimath
isfimathlocal(a)
ans =
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: FullPrecision
4-17
4 Working with fimath Objects
ans =
0
a = fi(pi);
b = fi(8);
isequal(a.fimath, b.fimath)
ans =
a + b
ans =
11.1416
To perform arithmetic with +, -, .*, or *, two fi operands must also have the
same data type. For example, you can add two fi objects with data type
double, but you can not add an object with data type double and one with
data type single:
a =
DataTypeMode: Double
4-18
fimath Properties Usage for Fixed-Point Arithmetic
b =
27
DataTypeMode: Double
a + b
ans =
30
DataTypeMode: Double
c =
12
DataTypeMode: Single
a + c
Math operations are not allowed on FI objects with different data types.
Fixed-point fi object operands do not have to have the same scaling. You can
perform binary math operations on a fi object with a fixed-point data type
and a fi object with a scaled doubles data type. In this sense, the scaled
double data type acts as a fixed-point data type:
a = fi(pi)
a =
3.1416
4-19
4 Working with fimath Objects
Signedness: Signed
WordLength: 16
FractionLength: 13
b = fi(magic(2), ...
'DataTypeMode', 'Scaled double: binary point scaling')
b =
1 3
4 2
a + b
ans =
4.1416 6.1416
7.1416 5.1416
Use the divide function to perform division with doubles, singles, or binary
point-only scaling fi objects.
4-20
fimath Properties Usage for Fixed-Point Arithmetic
f =
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: SpecifyPrecision
SumWordLength: 32
SumFractionLength: 16
CastBeforeSum: true
a = fi(pi, 'fimath', f)
a =
3.1416
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: SpecifyPrecision
SumWordLength: 32
4-21
4 Working with fimath Objects
SumFractionLength: 16
CastBeforeSum: true
b =
22
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: SpecifyPrecision
SumWordLength: 32
SumFractionLength: 16
CastBeforeSum: true
a + b
ans =
25.1416
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: SpecifyPrecision
SumWordLength: 32
SumFractionLength: 16
4-22
fimath Properties Usage for Fixed-Point Arithmetic
CastBeforeSum: true
4-23
4 Working with fimath Objects
p = fipref('NumberDisplay', 'RealWorldValue',...
'NumericTypeDisplay', 'none', 'FimathDisplay', 'none');
T = numerictype('WordLength',8,'FractionLength',7);
F = fimath('RoundingMethod','Floor','OverflowAction','Wrap');
a = fi(1,T,F)
a =
-1
b = fi(1,T)
b =
0.9922
Because you create a with a fimath object F that has OverflowAction set to
Wrap, the value of a wraps to -1. Conversely, because you create b with the
default OverflowAction value of Saturate, its value saturates to 0.9922.
b.fimath = F
b =
0.9922
4-24
fimath for Rounding and Overflow Modes
For more information about the fimath object and its properties, see “fimath
Object Properties” on page 4-6
4-25
4 Working with fimath Objects
Sharing fimath properties across fi objects ensures that the fi objects are
using the same arithmetic rules and helps you avoid “mismatched fimath”
errors.
a = fi(pi)
• Create a fi object using the sfi or ufi constructor. All fi objects created
with these constructors have no local fimath.
b = sfi(pi)
4-26
fimath for Sharing Arithmetic Rules
numerictype object and fimath object, and then uses those objects to create
two fi objects with the same numerictype and fimath attributes:
T = numerictype('WordLength',32,'FractionLength',30)
T =
F = fimath('RoundingMethod','Floor',...
'OverflowAction','Wrap')
F =
RoundingMethod: Floor
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
a = fi(pi, T, F)
a =
-0.8584
RoundingMethod: Floor
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
4-27
4 Working with fimath Objects
b = fi(pi/2, T, F)
b =
1.5708
RoundingMethod: Floor
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
4-28
fimath ProductMode and SumMode
Example Setup
The examples in the sections of this topic show the differences among the four
settings of the ProductMode and SumMode properties:
• FullPrecision
• KeepLSB
• KeepMSB
• SpecifyPrecision
p = fipref;
p.NumericTypeDisplay = 'short';
p.FimathDisplay = 'none';
p.LoggingMode = 'on';
F = fimath('OverflowAction','Wrap',...
'RoundingMethod','Floor',...
'CastBeforeSum',false);
warning off
format compact
Next, define fi objects a and b. Both have signed 8-bit data types. The
fraction length gets chosen automatically for each fi object to yield the best
possible precision:
a = fi(pi, true, 8)
4-29
4 Working with fimath Objects
a =
3.1563
s8,5
b = fi(exp(1), true, 8)
b =
2.7188
s8,5
FullPrecision
Now, set ProductMode and SumMode for a and b to FullPrecision and look
at some results:
F.ProductMode = 'FullPrecision';
F.SumMode = 'FullPrecision';
a.fimath = F;
b.fimath = F;
a
a =
3.1563 %011.00101
s8,5
b =
2.7188 %010.10111
s8,5
a*b
ans =
8.5811 %001000.1001010011
s16,10
a+b
ans =
5.8750 %0101.11100
4-30
fimath ProductMode and SumMode
s9,5
In FullPrecision mode, the product word length grows to the sum of the
word lengths of the operands. In this case, each operand has 8 bits, so the
product word length is 16 bits. The product fraction length is the sum of the
fraction lengths of the operands, in this case 5 + 5 = 10 bits.
The sum word length grows by one most significant bit to accommodate the
possibility of a carry bit. The sum fraction length aligns with the fraction
lengths of the operands, and all fractional bits are kept for full precision. In
this case, both operands have 5 fractional bits, so the sum has 5 fractional bits.
KeepLSB
Now, set ProductMode and SumMode for a and b to KeepLSB and look at some
results:
F.ProductMode = 'KeepLSB';
F.ProductWordLength = 12;
F.SumMode = 'KeepLSB';
F.SumWordLength = 12;
a.fimath = F;
b.fimath = F;
a
a =
3.1563 %011.00101
s8,5
b =
2.7188 %010.10111
s8,5
a*b
ans =
0.5811 %00.1001010011
s12,10
4-31
4 Working with fimath Objects
a+b
ans =
5.8750 %0000101.11100
s12,5
In KeepLSB mode, you specify the word lengths and the least significant bits
of results are automatically kept. This mode models the behavior of integer
operations in the C language.
The product fraction length is the sum of the fraction lengths of the operands.
In this case, each operand has 5 fractional bits, so the product fraction length
is 10 bits. In this mode, all 10 fractional bits are kept. Overflow occurs
because the full-precision result requires 6 integer bits, and only 2 integer
bits remain in the product.
The sum fraction length aligns with the fraction lengths of the operands, and
in this model all least significant bits are kept. In this case, both operands
had 5 fractional bits, so the sum has 5 fractional bits. The full-precision result
requires 4 integer bits, and 7 integer bits remain in the sum, so no overflow
occurs in the sum.
KeepMSB
Now, set ProductMode and SumMode for a and b to KeepMSB and look at some
results:
F.ProductMode = 'KeepMSB';
F.ProductWordLength = 12;
F.SumMode = 'KeepMSB';
F.SumWordLength = 12;
a.fimath = F;
b.fimath = F;
a
a =
3.1563 %011.00101
s8,5
4-32
fimath ProductMode and SumMode
b =
2.7188 %010.10111
s8,5
a*b
ans =
8.5781 %001000.100101
s12,6
a+b
ans =
5.8750 %0101.11100000
s12,8
In KeepMSB mode, you specify the word lengths and the most significant
bits of sum and product results are automatically kept. This mode models
the behavior of many DSP devices where the product and sum are kept in
double-wide registers, and the programmer chooses to transfer the most
significant bits from the registers to memory after each operation.
The full-precision product requires 6 integer bits, and the fraction length of
the product is adjusted to accommodate all 6 integer bits in this mode. No
overflow occurs. However, the full-precision product requires 10 fractional
bits, and only 6 are available. Therefore, precision is lost.
The full-precision sum requires 4 integer bits, and the fraction length of
the sum is adjusted to accommodate all 4 integer bits in this mode. The
full-precision sum requires only 5 fractional bits; in this case there are 8, so
there is no loss of precision.
This example shows that, in KeepMSB mode the fraction length changes
regardless of whether an overflow occurs. The fraction length is set to the
amount needed to represent the product in case both terms use the maximum
possible value (18+18-16=20 in this example).
F = fimath('SumMode','KeepMSB','ProductMode','KeepMSB',...
'ProductWordLength',16,'SumWordLength',16);
a=fi(100,1,16,-2,'fimath',F);
a*a
4-33
4 Working with fimath Objects
ans =
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: KeepMSB
ProductWordLength: 16
SumMode: KeepMSB
SumWordLength: 16
CastBeforeSum: true
SpecifyPrecision
Now set ProductMode and SumMode for a and b to SpecifyPrecision and
look at some results:
F.ProductMode = 'SpecifyPrecision';
F.ProductWordLength = 8;
F.ProductFractionLength = 7;
F.SumMode = 'SpecifyPrecision';
F.SumWordLength = 8;
F.SumFractionLength = 7;
a.fimath = F;
b.fimath = F;
a
a =
3.1563 %011.00101
s8,5
b =
2.7188 %010.10111
s8,5
4-34
fimath ProductMode and SumMode
a*b
ans =
0.5781 %0.1001010
s8,7
a+b
ans =
-0.1250 %1.1110000
s8,7
In SpecifyPrecision mode, you must specify both word length and fraction
length for sums and products. This example unwisely uses fractional formats
for the products and sums, with 8-bit word lengths and 7-bit fraction lengths.
The full-precision product requires 6 integer bits, and the example specifies
only 1, so the product overflows. The full-precision product requires 10
fractional bits, and the example only specifies 7, so there is precision loss in
the product.
The full-precision sum requires 4 integer bits, and the example specifies only
1, so the sum overflows. The full-precision sum requires 5 fractional bits, and
the example specifies 7, so there is no loss of precision in the sum.
For more information about the fimath object and its properties, see “fimath
Object Properties” on page 4-6
4-35
4 Working with fimath Objects
4-36
How Functions Use fimath
Functions Note
4-37
4 Working with fimath Objects
4-38
5
P = fipref
P =
NumberDisplay: 'RealWorldValue'
NumericTypeDisplay: 'full'
FimathDisplay: 'full'
LoggingMode: 'Off'
DataTypeOverride: 'ForceOff'
The syntax
P = fipref(...'PropertyName','PropertyValue'...)
allows you to set properties for a fipref object at object creation with property
name/property value pairs.
5-2
fipref Object Properties
These properties are described in detail in the “fi Object Properties” on page
2-19. To learn how to specify properties for fipref objects in Fixed-Point
Designer software, refer to “fipref Object Properties Setting” on page 5-3.
P =
NumberDisplay: 'bin'
NumericTypeDisplay: 'short'
5-3
5 Working with fipref Objects
FimathDisplay: 'full'
LoggingMode: 'Off'
DataTypeOverride: 'ForceOff'
P.NumberDisplay
ans =
bin
P.NumericTypeDisplay = 'full'
P =
NumberDisplay: 'bin'
NumericTypeDisplay: 'full'
FimathDisplay: 'full'
LoggingMode: 'Off'
DataTypeOverride: 'ForceOff'
5-4
fi Object Display Preferences Using fipref
For example, the following code shows the default fipref display for a fi
object with a local fimath object:
a =
3.1415
RoundingMethod: Floor
OverflowAction: Wrap
ProductMode: FullPrecision
SumMode: FullPrecision
The default fipref display for a fi object with no local fimath is as follows:
a = fi(pi)
a =
3.1416
P = fipref;
5-5
5 Working with fipref Objects
P.NumberDisplay = 'bin';
P.NumericTypeDisplay = 'short';
P.FimathDisplay = 'none'
P =
NumberDisplay: 'bin'
NumericTypeDisplay: 'short'
FimathDisplay: 'none'
LoggingMode: 'Off'
DataTypeOverride: 'ForceOff'
a
a =
0110010010000111
s16,13
For more information on the default fipref display, see “View Fixed-Point
Data”.
5-6
Underflow and Overflow Logging Using fipref
a = fi(1:5,1,8,6);
2 Define the fimath object associated with a, and indicate that you will
specify the sum and product word and fraction lengths.
F = a.fimath;
F.SumMode = 'SpecifyPrecision';
F.ProductMode = 'SpecifyPrecision';
a.fimath = F;
3 Define the fipref object and turn on overflow and underflow logging.
P = fipref;
P.LoggingMode = 'on';
P.NumericTypeDisplay = 'none';
P.FimathDisplay = 'none';
a.SumWordLength = 16;
a.SumFractionLength = 15;
5-7
5 Working with fipref Objects
a.ProductWordLength = 16;
a.ProductFractionLength = 15;
a(1) = pi
Warning: 1 overflow occurred in the fi assignment operation.
a =
a(1) = double(eps(a))/10
Warning: 1 underflow occurred in the fi assignment operation.
a =
a+a
Warning: 12 overflows occurred in the fi + operation.
ans =
a-a
Warning: 8 overflows occurred in the fi - operation.
ans =
0 0 0 0 0
5-8
Underflow and Overflow Logging Using fipref
a.*a
Warning: 4 product overflows occurred in the fi .* operation.
ans =
a*a'
Warning: 4 product overflows occurred in the fi * operation.
Warning: 3 sum overflows occurred in the fi * operation.
ans =
1.0000
Because overflows and underflows are logged as warnings, you can use the
dbstop MATLAB function with the syntax
dbstop if warning
to find the exact lines in a file that are causing overflows or underflows.
Use
5-9
5 Working with fipref Objects
For example, consider the following. First turn logging on, then perform
operations, and then finally get information about the operations:
fipref('LoggingMode','on');
x = fi([-1.5 eps 0.5], true, 16, 15);
x(1) = 3.0;
maxlog(x)
ans =
1.0000
minlog(x)
ans =
-1
noverflows(x)
ans =
nunderflows(x)
ans =
5-10
Underflow and Overflow Logging Using fipref
Next, reset the log and request the same information again. Note that the
functions return empty [], because logging has been reset since the operations
were run:
resetlog(x)
maxlog(x)
ans =
[]
minlog(x)
ans =
[]
noverflows(x)
ans =
[]
nunderflows(x)
ans =
[]
5-11
5 Working with fipref Objects
p = fipref('DataTypeOverride', 'TrueDoubles')
p =
NumberDisplay: 'RealWorldValue'
NumericTypeDisplay: 'full'
FimathDisplay: 'full'
LoggingMode: 'Off'
DataTypeOverride: 'TrueDoubles'
a = fi(pi)
a =
3.1416
DataTypeMode: Double
p = fipref('DataTypeOverride', 'ForceOff')
p =
5-12
Data Type Override Preferences Using fipref
NumberDisplay: 'RealWorldValue'
NumericTypeDisplay: 'full'
FimathDisplay: 'full'
LoggingMode: 'Off'
DataTypeOverride: 'ForceOff'
a =
3.1416
DataTypeMode: Double
b = fi(pi)
b =
3.1416
Tip To reset the fipref object to its default values use reset(fipref) or
reset(p), where p is a fipref object. This is useful to ensure that data type
override and logging are off.
5-13
5 Working with fipref Objects
4 Use the maxlog and minlog functions to log the maximum and minimum
values achieved by the variables in your algorithm in floating-point mode.
6 Use the information obtained in step 4 to set the fixed-point scaling for each
variable in your algorithm such that the full numerical range of each variable
is representable by its data type and scaling.
5-14
6
• You can use the numerictype constructor function to create a new object.
• You can use the numerictype constructor function to copy an existing
numerictype object.
T = numerictype
T =
6-2
numerictype Object Construction
This behavior differs from the behavior that occurs when you use a syntax
such as T = numerictype(s,w), where you only specify the property values
in the constructor. Using such a syntax results in no default numerictype
object being created, and the numerictype object receives only the assigned
property values that are specified in the constructor.
The following example shows how the property name/property value syntax
creates a slightly different numerictype object than the property values
syntax, even when you specify the same property values in both constructors.
T1 = numerictype('Signed',0,'WordLength',32)
T1 =
6-3
6 Working with numerictype Objects
Now, create another unsigned 32 bit numerictype object, but this time specify
only property values in the constructor.
T2 = numerictype(0,32)
T2 =
Unlike T1, T2 only has the property values you specified. The DataTypeMode
of T2 is Fixed-Point: unspecified scaling, so no fraction length is
assigned.
T = numerictype;
U = T;
isequal(T,U)
ans =
6-4
numerictype Object Construction
1 On the Home tab, in the File section, click New > Script to open the
MATLAB Editor
2 On the Editor tab, in the Edit section, click in the Insert button
group. Click the Insert numerictype... to open the Insert numerictype
Constructor dialog box.
3 Use the edit boxes and drop-down menus to specify the following properties
of the numerictype object:
• Data type mode = Fixed-point: binary point scaling
• Signedness = Signed
• Word length = 32
• Fraction length = 30
6-5
6 Working with numerictype Objects
6-6
numerictype Object Properties
6-7
6 Working with numerictype Objects
6-8
numerictype Object Properties
Scaling Scaling mode of the object • BinaryPoint (default) — Scaling for the fi
object is defined by the fraction length.
• SlopeBias — Scaling for the fi object is
defined by the slope and bias.
• Unspecified — A temporary setting that is
only allowed at fi object creation, to allow for
the automatic assignment of a binary point
best-precision scaling.
Signed Whether the object is signed • true (default) — signed
• false — unsigned
Note Although the Signed • 1 — signed
property is still supported, the
• 0 — unsigned
Signedness property always
appears in the numerictype • [] — auto
object display. If you choose to
change or set the signedness
of your numerictype objects
using the Signed property,
MATLAB updates the
corresponding value of the
Signedness property.
6-9
6 Working with numerictype Objects
Slope Slope adjustment associated • Any number greater than or equal to 1 and
with the object
AdjustmentFactor less than 2
The slope adjustment is
equivalent to the fractional
Note
slope of a fixed-point number.
slope = slope adjustment factor × 2 fixed exponent
Changing one of these properties changes the
other.
6-10
numerictype Object Properties
real-world value =
stored integer value × (slope adjustment factor × 2 fixed exponent ) + bias
which is equivalent to
If any of these properties are updated, the others are modified accordingly.
6-11
6 Working with numerictype Objects
For example, to set the word length to 32 bits and the fraction length to 30 bits,
T =
In addition to creating a numerictype object at the command line, you can also
set numerictype properties using the Insert numerictype Constructor
dialog box. For an example of this approach, see “Example: Build numerictype
Object Constructors in a GUI” on page 6-5.
T.WordLength
ans =
32
6-12
numerictype Object Properties
T.FractionLength = 31
T =
6-13
6 Working with numerictype Objects
Note When you create a fi object, any unspecified field of the numerictype
object reverts to its default value. Thus, if the DataTypeMode is set to
unspecified scaling, it defaults to binary point scaling when the fi
object is created. If the Signedness property of the numerictype object is set
to Auto, it defaults to Signed when the fi object is created.
Word- Fraction-
DataTypeMode DataType Scaling Signedness Length Length Slope Bias
Fixed-point data types
Fixed-point: Fixed BinaryPoint Signed Positive Positive 2^(-fraction
0
binary point Unsigned integer or length)
scaling Auto from negative
1 to integer
65,536
Fixed-point: Fixed SlopeBias Signed Positive N/A Any Any
slope and Unsigned integer floating- floating-
bias scaling Auto from point point
1 to number number
65,536
6-14
numerictype of Fixed-Point Objects
Word- Fraction-
DataTypeMode DataType Scaling Signedness Length Length Slope Bias
Fixed-point: Fixed Unspecified Signed Positive N/A N/A N/A
unspecified Unsigned integer
scaling Auto from
1 to
65,536
Scaled double data types
Scaled ScaledDouble BinaryPoint Signed Positive Positive 2^(-fraction
0
double: Unsigned integer or length)
binary point Auto from negative
scaling 1 to integer
65,536
Scaled ScaledDouble SlopeBias Signed Positive N/A Any Any
double: Unsigned integer floating- floating-
slope and Auto from point point
bias scaling 1 to number number
65,536
Scaled ScaledDouble Unspecified Signed Positive N/A N/A N/A
double: Unsigned integer
unspecified Auto from
scaling 1 to
65,536
Built-in data types
Double double N/A 1 64 0 1 0
true
Single single N/A 1 32 0 1 0
true
Boolean boolean N/A 0 1 0 1 0
false
6-15
6 Working with numerictype Objects
real-world value =
stored integer value × (slope adjustment factor × 2 fixed exponent ) + bias
which is equivalent to
If any of these properties are updated, the others are modified accordingly.
6-16
numerictype Objects Usage to Share Data Type and Scaling Settings of fi objects
Example 1
In the following example, you create a numerictype object T with word length
32 and fraction length 28. Next, to ensure that your fi objects have the same
numerictype attributes, create fi objects a and b using your numerictype
object T.
format long g
T = numerictype('WordLength',32,'FractionLength',28)
T =
a = fi(pi,T)
a =
3.1415926553309
b = fi(pi/2, T)
b =
6-17
6 Working with numerictype Objects
1.5707963258028
Example 2
In this example, start by creating a numerictype object T with [Slope Bias]
scaling. Next, use that object to create two fi objects, c and d with the same
numerictype attributes:
T =
c = fi(pi, T)
c =
d = fi(pi/2, T)
d =
6-18
numerictype Objects Usage to Share Data Type and Scaling Settings of fi objects
6-19
6 Working with numerictype Objects
6-20
7
• You can use the quantizer constructor function to create a new object.
• You can use the quantizer constructor function to copy a quantizer object.
q = quantizer
q =
DataMode = fixed
RoundingMethod = Floor
OverflowAction = Saturate
Format = [16 15]
q = quantizer;
r = q;
isequal(q,r)
ans =
A listing of all the properties of the quantizer object q you just created is
displayed along with the associated property values. All property values
are set to defaults when you construct a quantizer object this way. See
“quantizer Object Properties” on page 7-3 for more details.
7-2
quantizer Object Properties
See the“fi Object Properties” on page 2-19 for more details about these
properties, including their possible values.
type
q = quantizer('datamode','fixed','format',[16,14],...
'OverflowMode','saturate','RoundMode','ceil')
You do not have to include quantizer object property names when you set
quantizer object property values.
For example, you can create quantizer object q from the previous example
by typing
q = quantizer('fixed',[16,14],'saturate','ceil')
Note You do not have to include default property values when you construct
a quantizer object. In this example, you could leave out 'fixed' and
'saturate'.
7-3
7 Working with quantizer Objects
Once you quantize data with a quantizer object, its state values might
change.
format long g
rng('default');
x = randn(100,4);
q = quantizer([16,14]);
q.maxlog
ans =
-1.79769313486232e+308
q.noverflows
ans =
Note that maxlog is equal to -realmax, which indicates that the quantizer
q is in a reset state.
7-4
Quantizing Data with quantizer Objects
y = quantize(q,x);
Warning: 626 overflow(s) occurred in the fi quantize operation.
q.maxlog
ans =
1.99993896484375
q.noverflows
ans =
626
Note that the maximum logged value was taken after quantization, that is,
q.maxlog == max(y).
reset(q)
q.maxlog
ans =
-1.79769313486232e+308
q.noverflows
ans =
7-5
7 Working with quantizer Objects
Use
For example,
q = quantizer([3 2]);
x = [0.75 -0.25
0.50 -0.50
0.25 -0.75
0 -1 ];
b = num2bin(q,x)
b =
011
010
001
000
111
110
101
100
7-6
8
Fixed-Point Conversion
You can convert floating-point MATLAB code to fixed point manually or, if
you have a MATLAB Coder™ license, you can use the automated fixed-point
conversion tool to convert from floating-point MATLAB code to fixed-point
C code.
Manual Workflow
If you have a baseline understanding of fixed-point implementation details
and an interest in exploring design tradeoffs to achieve optimized results, use
the separate algorithm/data type workflow. Separating algorithmic code from
data type specifications allows you to quickly explore design tradeoffs. This
approach provides readable, portable fixed-point code that you can easily
integrated into other projects. For more information, see “Manual Fixed-Point
Conversion Workflow” on page 8-87 and “Implement FIR Filter Algorithm for
Floating-Point and Fixed-Point Types using cast and zeros” on page 8-34.
8-2
Automated Fixed-Point Conversion
License Requirements
Fixed-point conversion requires the following licenses:
• Fixed-Point Designer
• MATLAB Coder
You can manually enter static ranges. These manually-entered ranges take
precedence over simulation ranges and the tool uses them when proposing
data types. In addition, you can modify and lock the proposed type so that
8-3
8 Fixed-Point Conversion
the tool cannot change it. For more information, see “Locking Proposed Data
Types” on page 8-11.
8-4
Automated Fixed-Point Conversion
• Verify that your test files cover the full intended operating range of your
algorithm using code coverage results.
• Propose fraction lengths based on default word lengths.
• Propose word lengths based on default fraction lengths.
• Optimize whole numbers.
8-5
8 Fixed-Point Conversion
Code Coverage
By default, the Fixed-Point Conversion tool shows code coverage results. Your
test files should exercise the algorithm over its full operating range so that
the simulation ranges are accurate. The quality of the proposed fixed-point
data types depends on how well the test files cover the operating range of the
algorithm with the accuracy that you want. Reviewing code coverage results
helps you verify that your test file is exercising the algorithm adequately. If
the code coverage is inadequate, modify the test file or add more test files
to increase coverage. If you simulate multiple test files in one run, the tool
displays cumulative coverage. However, if you specify multiple test files but
run them one at a time, the tool displays the coverage of the file that ran last.
Code coverage is on by default. Turn it off only after you have verified that
you have adequate test file coverage. Turning off code coverage might speed
up simulation. To turn off code coverage, in the Fixed-Point Conversion tool:
The tool covers basic MATLAB control constructs and shows statement
coverage for basic blocks of code. The tool displays a color-coded coverage bar
to the left of the code.
8-6
Automated Fixed-Point Conversion
When you position your cursor over the coverage bar, the color highlighting
extends over the code and the tool displays more information about how
8-7
8 Fixed-Point Conversion
often the code is executed. For MATLAB constructs that affect control flow
(if-elseif-else, switch-case, for-continue-break, return), it displays statement
coverage as a percentage coverage for basic blocks inside these constructs.
To verify that your test file is testing your algorithm over the intended
operating range, review the code coverage results and take action as described
in the following table.
8-8
Automated Fixed-Point Conversion
Note You cannot propose data types based on derived ranges for MATLAB
classes.
You can manually enter static ranges. These manually-entered ranges take
precedence over simulation ranges and the tool uses them when proposing
data types. In addition, you can modify and lock the proposed type so that
8-9
8 Fixed-Point Conversion
the tool cannot change it. For more information, see “Locking Proposed Data
Types” on page 8-11.
Running a Simulation
When you open the Fixed-Point Conversion tool, the tool generates an
instrumented MEX function for your entry-point MATLAB file. If the build
completes without errors, the tool displays compiled information (type,
size, complexity) for functions and variables in your code. To navigate to
local functions, click the Functions tab. If build errors occur, the tool
provides error messages that link to the line of code that caused the build
issues. You must address these errors before running a simulation. Use the
link to navigate to the offending line of code in the MATLAB editor and
modify the code to fix the issue. If your code uses functions that are not
supported for fixed-point conversion, the tool displays them on the Function
Replacements tab. See “Function Replacements” on page 8-17.
Before running a simulation, specify the test file or files that you want to
run. When you run a simulation, the tool runs the test file, calling the
instrumented MEX function. If you modify the MATLAB design code, the tool
automatically generates an updated MEX function before running a test file.
If the test file runs successfully, the simulation minimum and maximum
values and the proposed types are displayed on the Variables tab. If you
manually enter static ranges for a variable, the manually-entered ranges take
precedence over the simulation ranges. If you manually modify the proposed
types by typing or using the histogram, the data types are locked so that
the tool cannot modify them.
If the test file fails, the errors are displayed on the Simulation Output tab.
Test files should exercise your algorithm over its full operating range. The
quality of the proposed fixed-point data types depends on how well the test file
covers the operating range of the algorithm with the desired accuracy. You can
add test files and select to run more than one test file during the simulation. If
you run multiple test files, the conversion tool merges the simulation results.
Optionally, you can select to log data for histograms. After running a
simulation, you can view the histogram for each variable. For more
information, see “Histogram” on page 8-15.
8-10
Automated Fixed-Point Conversion
To compute derived ranges and propose data types based on these ranges,
provide static minimum and maximum values or proposed data types for
all input variables. To improve the analysis, enter as much static range
information as possible for other variables. You can manually enter ranges or
promote simulation ranges to use as static ranges. Manually-entered static
ranges always take precedence over simulation ranges.
If you know what data type your hardware target uses, set the proposed data
types to match this type. Manually-entered data types are locked so that the
tool cannot modify them. The tool uses these data types to calculate the input
minimum and maximum values and to derive ranges for other variables. For
more information, see “Locking Proposed Data Types” on page 8-11.
When you select Compute Derived Ranges, the tool runs a derived range
analysis to compute static ranges for variables in your MATLAB algorithm.
When the analysis is complete, the static ranges are displayed on the
Variables tab. If the run produces +/-Inf derived ranges, consider defining
ranges for all persistent variables.
Optionally, you can select Quick derived range analysis. With this option,
the conversion tool performs faster static analysis. The computed ranges
might be larger than necessary. Select this option in cases where the static
analysis takes more time than you can afford.
If the derived range analysis for your project is taking a long time, you can
optionally set a timeout. The tool aborts the analysis when the timeout is
reached.
8-11
8 Fixed-Point Conversion
The tool displays locked data types in bold so that they are easy to identify.
You can unlock a type using one of the following methods:
Viewing Functions
You can view a list of functions in your project on the Navigation pane. This
list also includes function specializations and class methods. When you select
a function from the list, the MATLAB code for that function or class method is
displayed in the Fixed-Point Conversion tool code window.
Viewing Variables
The Variables tab provides the following information for each variable in the
function selected in the Navigation pane:
• Type — The original data type of the variable in the MATLAB algorithm.
• Sim Min and Sim Max — The minimum and maximum values assigned
to the variable during simulation.
You can edit the simulation minimum and maximum values. Edited
fields are shown in bold. Editing these fields does not trigger static range
analysis, but the tool uses the edited values in subsequent analyses. You
can revert to the types proposed by the tool.
• Static Min and Static Max — The static minimum and maximum values.
8-12
Automated Fixed-Point Conversion
To compute derived ranges and propose data types based on these ranges,
provide static minimum and maximum values for all input variables. To
improve the analysis, enter as much static range information as possible
for other variables.
When you compute derived ranges, the Fixed-Point Conversion tool runs a
static analysis to compute static ranges for variables in your code. When
the analysis is complete, the static ranges are displayed. You can edit the
computed results. Edited fields are shown in bold. Editing these fields does
not trigger static range analysis, but the tool uses the edited values in
subsequent analyses. You can revert to the types proposed by the tool.
• Whole Number — Whether all values assigned to the variable during
simulation are integers.
The Fixed-Point Conversion tool determines whether a variable is always
a whole number. You can modify this field. Edited fields are shown in
bold. Editing these fields does not trigger static range analysis, but the
tool uses the edited values in subsequent analyses. You can revert to the
types proposed by the tool.
• The proposed fixed-point data type for the specified word (or fraction)
length. Proposed data types use the numerictype notation. For example,
numerictype(1,16,12) denotes a signed fixed-point type with a word
length of 16 and a fraction length of 12. numerictype(0,16,12) denotes an
unsigned fixed-point type with a word length of 16 and a fraction length
of 12.
Because the tool does not apply data types to expressions, it does not display
proposed types for them. Instead, it displays their original data types.
You can also view and edit variable information in the code pane by placing
your cursor over a variable name.
You can use Ctrl+F to search for variables in the MATLAB code and on the
Variables tab. The tool highlights occurrences in the code and displays only
the variable with the specified name on the Variables tab.
8-13
8 Fixed-Point Conversion
• Code for MATLAB classes and code coverage for class methods in the code
window. Use the Function list in the Navigation bar to select which class
or class method to view.
8-14
Automated Fixed-Point Conversion
Histogram
To log data for histograms, in the Fixed-Point Conversion window, click
Run Simulation and select Log data for histogram, and then click the
Run Simulation button.
After simulation, to view the histogram for a variable, on the Variables tab,
click the Proposed Type field for that variable.
The histogram provides the range of the proposed data type and the
percentage of simulation values that the proposed data type covers. The bit
weights are displayed along the X-axis, and the percentage of occurrences
along the Y-axis. Each bin in the histogram corresponds to a bit in the binary
word. For example, this histogram displays the range for a variable of type
numerictype(1,16,14).
8-15
8 Fixed-Point Conversion
You can view the effect of changing the proposed data types by:
• Dragging the edges of the bounding box in the histogram window to change
the proposed data type.
8-16
Automated Fixed-Point Conversion
Function Replacements
If your MATLAB code uses functions that do not have fixed-point support,
the tool lists these functions on the Function Replacements tab. You can
add and remove function replacements from this list. If you enter a function
replacements for a function, the replacement function is used when you build
the project. If you do not enter a replacement, the tool uses the type specified
in the original MATLAB code for the function.
Note Using this table, you can replace the names of the functions but you
cannot replace argument patterns.
Validating Types
Selecting Validate Types validates the build using the proposed fixed-point
data types. If the validation is successful, you are ready to test the numerical
behavior of the fixed-point MATLAB algorithm.
If the errors or warnings occur during validation, they are displayed on the
Type Validation Output tab. If errors or warning occur:
• On the Variables tab, inspect the proposed types and manually modified
types to verify that they are valid.
• On the Function Replacements tab, verify that you have provided
function replacements for unsupported functions.
Testing Numerics
After validating the proposed fixed-point data types, select Test Numerics to
verify the behavior of the fixed-point MATLAB algorithm. By default, if you
added a test file to define inputs or run a simulation, the tool uses this test
8-17
8 Fixed-Point Conversion
file to test numerics. Optionally, you can add test files and select to run more
than one test file. The tool compares the numerical behavior of the generated
fixed-point MATLAB code with the original floating-point MATLAB code. If
you select to log inputs and outputs for comparison plots, the tool generates an
additional plot for each scalar output. This plot shows the floating-point and
fixed-point results and the difference between them. For non-scalar outputs,
only the error information is shown.
If the numerical results do not meet your desired accuracy after fixed-point
simulation, modify fixed-point data type settings and repeat the type
validation and numerical testing steps. You might have to iterate through
these steps multiple times to achieve the desired results.
Detecting Overflows
When testing numerics, selecting Use scaled doubles to detect overflows
enables overflow detection. When this option is selected, the conversion
tool runs the simulation using scaled double versions of the proposed
fixed-point types. Because scaled doubles store their data in double-precision
floating-point, they carry out arithmetic in full range. They also retain their
fixed-point settings, so they are able to report when a computation goes out of
the range of the fixed-point type. For more information, see “Scaled Doubles”
on page 27-20.
8-18
Automated Fixed-Point Conversion
If your original algorithm uses scaled doubles, the tool also provides overflow
information for these expressions.
See Also
“Detect Overflows Using the Fixed-Point Conversion Tool” on page 8-23
8-19
8 Fixed-Point Conversion
8-20
Specify Type Proposal Options
Yes
8-21
8 Fixed-Point Conversion
8-22
Detect Overflows Using the Fixed-Point Conversion Tool
Prerequisites
• MATLAB
• MATLAB Coder
• Fixed-Point Designer
Create a Project
Add File
function y = overflow(b,x,reset)
if nargin<3, reset = true; end
persistent z p
if isempty(z) || reset
p = 0;
z = zeros(size(b));
end
[y,z,p] = fir_filter(b,x,z,p);
end
function [y,z,p] = fir_filter(b,x,z,p)
8-23
8 Fixed-Point Conversion
y = zeros(size(x));
nx = length(x);
nb = length(b);
for n=1:nx
p=p+1; if p>nb, p=1; end
z(p) = x(n);
acc = 0;
k = p;
for j=1:nb
acc = acc + b(j)*z(k);
k=k-1; if k<1, k=nb; end
end
y(n) = acc;
end
end
2 On the project Overview tab, click the Add files link. Browse to the file
overflow.m and then click OK to add the file to the project.
function overflow_test
% The filter coefficients were computed using the FIR1 function from
% Signal Processing Toolbox.
% b = fir1(11,0.25);
b = [-0.004465461051254
-0.004324228005260
+0.012676739550326
+0.074351188907780
+0.172173206073645
+0.249588554524763
+0.249588554524763
+0.172173206073645
+0.074351188907780
+0.012676739550326
-0.004324228005260
8-24
Detect Overflows Using the Fixed-Point Conversion Tool
-0.004465461051254]';
% Input signal
nx = 256;
t = linspace(0,10*pi,nx)';
% Impulse
x_impulse = zeros(nx,1); x_impulse(1) = 1;
% Max Gain
% The maximum gain of a filter will occur when the inputs line up wit
% signs of the filter's impulse response.
x_max_gain = sign(b)';
x_max_gain = repmat(x_max_gain,ceil(nx/length(b)),1);
x_max_gain = x_max_gain(1:nx);
% Sums of sines
f0=0.1; f1=2;
x_sines = sin(2*pi*t*f0) + 0.1*sin(2*pi*t*f1);
% Chirp
f_chirp = 1/16; % Target frequency
x_chirp = sin(pi*f_chirp*t.^2); % Linear chirp
for i=1:size(x,2)
reset = true;
y(:,i) = overflow(b,x(:,i),reset);
end
test_plot(1,titles,t,x,y)
end
function test_plot(fig,titles,t,x,y1)
figure(fig)
clf
8-25
8 Fixed-Point Conversion
sub_plot = 1;
font_size = 10;
for i=1:size(x,2)
subplot(4,1,sub_plot); sub_plot = sub_plot+1;
plot(t,x(:,i),'c',t,y1(:,i),'k')
axis('tight')
xlabel('t','FontSize',font_size);
title(titles{i},'FontSize',font_size);
set(gca,'FontSize',12);
end
figure(gcf)
end
3 In the Autodefine Input Types dialog box, add overflow_test as a test file
and then click Run.
The test file runs and displays the outputs of the filter for each of the input
signals.MATLAB Coder determines the input types from the test file and
then displays them.
4 In the Autodefine Input Types dialog box, click Use These Types.
Fixed-Point Conversion
The project indicates that you must first define the fixed-point data types.
8-26
Detect Overflows Using the Fixed-Point Conversion Tool
complexity — for variables in your code. For more information, see “View
and Modify Variable Information”.
4 In the Fixed-Point Conversion tool, click Run Simulation and verify that
the overflow_test file is selected as the test file to run and then click
the Run Simulation button.
The simulation runs and the conversion tool displays simulation minimum
and maximum ranges on the Variables tab. Using the simulation range
data, the software proposes fixed-point types for each variable based on the
default type proposal settings, and displays them in the Proposed Type
column. The Validate Types option is now enabled.
8-27
8 Fixed-Point Conversion
5 To validate the build using the proposed types, click Validate Types.
If the errors or warnings occur during validation, they are displayed on the
Type Validation Output tab. For more information, see “Validating
Types” on page 8-17.
The tool runs the test file that you used to define input types to test the
fixed-point MATLAB code. Because you selected to detect overflows, it also
runs the simulation using scaled double versions of the proposed fixed-point
types. Scaled doubles store their data in double-precision floating-point, so
they carry out arithmetic in full range. They also retain their fixed-point
settings, so they are able to report when a computation goes out of the
range of the fixed-point type.
The simulation runs and the tool detects an overflow which it reports on
the Overflow tab.
Overflow error in expression 'acc + b( j )*z( k )'. Percentage of Current Range = 103.52%.
2 Click the overflow message. The tool highlights the expression that
overflowed in the code window.
8-28
Detect Overflows Using the Fixed-Point Conversion Tool
acc + b( j )*z( k )
The overflow still occurs, indicating that it is the addition in the expression
that is overflowing.
8-29
8 Fixed-Point Conversion
Prerequisites
• MATLAB
• MATLAB Coder
• Fixed-Point Designer
function y = overflow(b,x,reset)
if nargin<3, reset = true; end
persistent z p
if isempty(z) || reset
p = 0;
z = zeros(size(b));
end
[y,z,p] = fir_filter(b,x,z,p);
end
function [y,z,p] = fir_filter(b,x,z,p)
y = zeros(size(x));
nx = length(x);
nb = length(b);
for n=1:nx
p=p+1; if p>nb, p=1; end
z(p) = x(n);
acc = 0;
k = p;
for j=1:nb
acc = acc + b(j)*z(k);
k=k-1; if k<1, k=nb; end
end
8-30
Detect Overflows at the Command Line
y(n) = acc;
end
end
function overflow_test
% The filter coefficients were computed using the FIR1 function from
% Signal Processing Toolbox.
% b = fir1(11,0.25);
b = [-0.004465461051254
-0.004324228005260
+0.012676739550326
+0.074351188907780
+0.172173206073645
+0.249588554524763
+0.249588554524763
+0.172173206073645
+0.074351188907780
+0.012676739550326
-0.004324228005260
-0.004465461051254]';
% Input signal
nx = 256;
t = linspace(0,10*pi,nx)';
% Impulse
x_impulse = zeros(nx,1); x_impulse(1) = 1;
% Max Gain
% The maximum gain of a filter will occur when the inputs line up with
% signs of the filter's impulse response.
x_max_gain = sign(b)';
x_max_gain = repmat(x_max_gain,ceil(nx/length(b)),1);
x_max_gain = x_max_gain(1:nx);
% Sums of sines
f0=0.1; f1=2;
8-31
8 Fixed-Point Conversion
% Chirp
f_chirp = 1/16; % Target frequency
x_chirp = sin(pi*f_chirp*t.^2); % Linear chirp
for i=1:size(x,2)
reset = true;
y(:,i) = overflow(b,x(:,i),reset);
end
test_plot(1,titles,t,x,y)
end
function test_plot(fig,titles,t,x,y1)
figure(fig)
clf
sub_plot = 1;
font_size = 10;
for i=1:size(x,2)
subplot(4,1,sub_plot); sub_plot = sub_plot+1;
plot(t,x(:,i),'c',t,y1(:,i),'k')
axis('tight')
xlabel('t','FontSize',font_size);
title(titles{i},'FontSize',font_size);
set(gca,'FontSize',12);
end
figure(gcf)
end
fixptcfg = coder.config('fixpt');
Set the test bench name. In this example, the test bench function name is
overflow_test.
8-32
Detect Overflows at the Command Line
fixptcfg.TestBenchName = 'overflow_test';
fixptcfg.DefaultWordLength = 16;
fixptcfg.TestNumerics = true;
fixptcfg.DetectFixptOverflows = true;
Set the fimath Product mode and Sum mode to KeepLSB. These settings
models the behavior of integer operations in the C language.
fixptcfg.fimath = ...
'fimath( ''RoundingMethod'', ''Floor'', ''OverflowAction'', ''Wrap'', ''ProductMode'', ''KeepLSB'', ''SumMod
cfg = coder.config('lib');
Overflow error in expression 'acc + b( j )*z( k )'. Percentage of Current Range = 103.52%.
fixptcfg.fimath = ...
'fimath(''RoundingMethod'',''Floor'',''OverflowAction'',''Wrap'',''ProductMode'',''FullPrecision'',''SumMode
codegen -float2fixed fixptcfg -config cfg overflow
The numerics testing phase still reports an overflow, indicating that it is the
addition in the expression that is overflowing.
8-33
8 Fixed-Point Conversion
Separating data type type specification from algorithm code allows you to:
Original Algorithm
This example converts MATLAB® code for a finite impulse response (FIR)
filter to fixed point.
The formula for the n’th output y(n) of an (FIR) filter, given filter coefficients
b, and input x is:
There are several different ways to write an FIR filter. One way is with a
linear buffer like in the following function, where b is a row vector and z is a
column vector the same length as b.
8-34
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
end
end
To implement the FIR filter more efficiently, you can store the
states in a circular buffer, z, whose elements are z(p) = x(n), where
p=mod(n-1,length(b))+1, for n=1, 2, 3, ....
p = 0, z = [ 0 0 0 ]
Start with the first sample and fill the state buffer z in a circular manner.
...
8-35
8 Fixed-Point Conversion
You can implement the FIR filter using a circular buffer like the following
MATLAB function.
Test File
The test vectors should represent realistic inputs that exercise the full range
of values expected by your system. Realistic inputs are impulses, sums of
sinusoids, and chirp signals, for which you can verify that the outputs are
correct using linear theory. Signals that produce maximum output are useful
for verifying that your system does not overflow.
Set up
Run the following code to capture and reset the current state of global
fixed-point math settings and fixed-point preferences.
resetglobalfimath;
FIPREF_STATE = get(fipref);
8-36
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
resetfipref;
Run the following code to copy the test functions into a temporary folder so
this example doesn’t interfere with your own work.
tempdirObj = fidemo.fiTempdir('fir_filt_circ_buff_fixed_point_conversion_ex
copyfile(fullfile(matlabroot,'toolbox','fixedpoint','fidemos','+fidemo',...
'fir_filt_*.m'),'.','f');
Filter coefficients
Use the following low-pass filter coefficients that were computed using the
fir1 function from Signal Processing Toolbox.
b = fir1(11,0.25);
b = [-0.004465461051254
-0.004324228005260
+0.012676739550326
+0.074351188907780
+0.172173206073645
+0.249588554524763
+0.249588554524763
+0.172173206073645
+0.074351188907780
+0.012676739550326
-0.004324228005260
-0.004465461051254]';
Time vector
nx = 256;
t = linspace(0,10*pi,nx)';
Impulse input
8-37
8 Fixed-Point Conversion
The maximum output of a filter occurs when the signs of the inputs line up
with the signs of the filter’s impulse response.
x_max_output = sign(fliplr(b))';
x_max_output = repmat(x_max_output,ceil(nx/length(b)),1);
x_max_output = x_max_output(1:nx);
The maximum magnitude of the output is the 1-norm of its impulse response,
which is norm(b,1) = sum(abs(b)).
maximum_output_magnitude =
1.0352
Sum of sines
A sum of sines is a typical input for a filter and you can easily see the high
frequencies filtered out in the plot.
f0=0.1; f1=2;
x_sines = sin(2*pi*t*f0) + 0.1*sin(2*pi*t*f1);
Chirp
A chirp gives a good visual of the low-pass filter action of passing the low
frequencies and attenuating the high frequencies.
8-38
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
Before starting the conversion to fixed point, call your original function with
the test file inputs to establish a baseline to compare to subsequent outputs.
y0 = zeros(size(x));
for i=1:size(x,2)
% Initialize the states for each column of input
p = 0;
z = zeros(size(b));
y0(:,i) = fir_filt_circ_buff_original(b,x(:,i),z,p);
end
Baseline Output
fir_filt_circ_buff_plot(1,titles,t,x,y0)
The first step after the algorithm works in MATLAB is to prepare it for
instrumentation, which requires code generation. Before the conversion,
you can use the coder.screener function to analyze your code and identify
unsupported functions and language features.
Entry-point function
function y = fir_filt_circ_buff_original_entry_point(b,x,reset)
if nargin<3, reset = true; end
8-39
8 Fixed-Point Conversion
Test file
function y = fir_filt_circ_buff_test(b,x)
y = zeros(size(x));
for i=1:size(x,2)
reset = true;
y(:,i) = fir_filt_circ_buff_original_entry_point_mex(b,x(:,i),rese
end
end
reset = true;
buildInstrumentedMex fir_filt_circ_buff_original_entry_point -args {b, x(:,
Run your test file inputs through the algorithm to log minimum and
maximum values.
y1 = fir_filt_circ_buff_test(b,x);
8-40
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
Show types
showInstrumentationResults fir_filt_circ_buff_original_entry_point_mex
Every time you modify your function, validate that the results still match
your baseline.
fir_filt_circ_buff_plot2(2,titles,t,x,y0,y1)
2 Modify the algorithm code to use data types from that table.
This example shows the iterative steps by creating different files. In practice,
you can make the iterative changes to the same file.
8-41
8 Fixed-Point Conversion
Create a types table using a structure with prototypes for the variables set
to their original types. Use the baseline types to validate that you made the
initial conversion correctly, and also use it to programatically toggle your
function between floating point and fixed point types. The index variables j,
k, n, nb, nx are automatically converted to integers by MATLAB Coder™, so
you don’t need to specify their types in the table.
Specify the prototype values as empty ([ ]) since the data types are used,
but not the values.
function T = fir_filt_circ_buff_original_types()
T.acc=double([]);
T.b=double([]);
T.p=double([]);
T.x=double([]);
T.y=double([]);
T.z=double([]);
end
The function call acc = cast(0,’like’,T.acc) casts the value 0 with the same
properties as variable T.acc. Initially, T.acc is a double defined in function
fir_filt_circ_buff_original_types, but it is re-defined as a fixed-point type later
in this example.
8-42
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
y = zeros(size(x),'like',T.y);
nx = length(x);
nb = length(b);
for n=1:nx
p(:)=p+1; if p>nb, p(:)=1; end
z(p) = x(n);
acc = cast(0,'like',T.acc);
k = p;
for j=1:nb
acc(:) = acc + b(j)*z(k);
k(:)=k-1; if k<1, k(:)=nb; end
end
y(n) = acc;
end
end
The function call p1 = cast(0,’like’,T1.p) casts the value 0 with the same
properties as variable T1.p. Initially, T1.p is a double defined in function
fir_filt_circ_buff_original_types, but it is re-defined as an integer type later
in this example.
function y1 = fir_filt_circ_buff_typed_entry_point(b,x,reset)
if nargin<3, reset = true; end
%
% Baseline types
%
T1 = fir_filt_circ_buff_original_types();
% Each call to the filter needs to maintain its own states.
persistent z1 p1
if isempty(z1) || reset
p1 = cast(0,'like',T1.p);
z1 = zeros(size(b),'like',T1.z);
end
8-43
8 Fixed-Point Conversion
b1 = cast(b,'like',T1.b);
x1 = cast(x,'like',T1.x);
[y1,z1,p1] = fir_filt_circ_buff_typed(b1,x1,z1,p1,T1);
end
Every time you modify your function, validate that the results still match
your baseline. Since you used the original types in the types table, the outputs
should be identical. This validates that you made the conversion to separate
the types from the algorithm correctly.
y1 = fir_filt_circ_buff_typed_test(b,x);
fir_filt_circ_buff_plot2(3,titles,t,x,y0,y1)
Use the proposed types from the Code Generation Report to guide you
in choosing fixed-point types and create a fixed-point types table using a
structure with prototypes for the variables.
8-44
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
Specify the prototype values as empty ([ ]) since the data types are used,
but not the values.
function T = fir_filt_circ_buff_fixed_point_types()
T.acc=fi([],true,32,30);
T.b=fi([],true,16,17);
T.p=int16([]);
T.x=fi([],true,16,14);
T.y=fi([],true,16,14);
T.z=fi([],true,16,14);
end
T2 = fir_filt_circ_buff_fixed_point_types();
persistent z2 p2
if isempty(z2) || reset
p2 = cast(0,'like',T2.p);
z2 = zeros(size(b),'like',T2.z);
end
b2 = cast(b,'like',T2.b);
x2 = cast(x,'like',T2.x);
[y2,z2,p2] = fir_filt_circ_buff_typed(b2,x2,z2,p2,T2);
[y1,y2] = fir_filt_circ_buff_typed_test(b,x);
showInstrumentationResults fir_filt_circ_buff_typed_entry_point_mex
8-45
8 Fixed-Point Conversion
[y2,z2,p2] = fir_filt_circ_buff_typed(b2,x2,z2,p2,T2);
Validate that the results are within an acceptable tolerance of your baseline.
fir_filt_circ_buff_plot2(4,titles,t,x,y1,y2);
Your algorithm has now been converted to fixed-point MATLAB code. If you
also want to convert to C-code, then proceed to the next section.
Generate C-Code
This section describes how to generate efficient C-code from the fixed-point
MATLAB code from the previous section.
Required products
You need MATLAB Coder™ to generate C-code, and you need Embedded
Coder® for the hardware implementation settings used in this example.
8-46
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
A rule of thumb for when to use coder.nullcopy is when the initialization takes
significant time compared to the rest of the algorithm. If you are not sure,
then the safest thing to do is to not use it.
You can set the fixed-point math properties to match the native actions of C.
This generates the most efficient C-code, but this example shows that it can
create problems with overflow and produce less accurate results which are
corrected in the next section. It doesn’t always create problems, though, so it
is worth trying first to see if you can get the cleanest possible C-code.
Set the fixed-point math properties to use floor rounding and wrap overflow
because those are the default actions in C.
Set the fixed-point math properties of products and sums to match native C
32-bit integer types, and to keep the least significant bits (LSBs) of math
operations.
8-47
8 Fixed-Point Conversion
function T = fir_filt_circ_buff_dsp_types()
F = fimath('RoundingMethod','Floor',...
'OverflowAction','Wrap',...
'ProductMode','KeepLSB',...
'ProductWordLength',32,...
'SumMode','KeepLSB',...
'SumWordLength',32);
T.acc=fi([],true,32,30,F);
T.p=int16([]);
T.b=fi([],true,16,17,F);
T.x=fi([],true,16,14,F);
T.y=fi([],true,16,14,F);
T.z=fi([],true,16,14,F);
end
Add a call to the types table in the entry-point function and run the test file.
In the second row of plots, you can see that the maximum output error is
twice the size of the input, indicating that a value that should have been
positive overflowed to negative. You can also see that the other outputs did
not overflow. This is why it is important to have your test file exercise the full
range of values in addition to other typical inputs.
fir_filt_circ_buff_plot2(5,titles,t,x,y1,y3);
8-48
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
Change the data types to scaled double, and add these settings to a
scaled-double types table.
function T = fir_filt_circ_buff_scaled_double_types()
F = fimath('RoundingMethod','Floor',...
'OverflowAction','Wrap',...
'ProductMode','KeepLSB',...
'ProductWordLength',32,...
'SumMode','KeepLSB',...
'SumWordLength',32);
DT = 'ScaledDouble';
T.acc=fi([],true,32,30,F,'DataType',DT);
T.p=int16([]);
T.b=fi([],true,16,17,F,'DataType',DT);
T.x=fi([],true,16,14,F,'DataType',DT);
T.y=fi([],true,16,14,F,'DataType',DT);
T.z=fi([],true,16,14,F,'DataType',DT);
end
Add a call to the the scaled-double types table to the entry-point function
and run the test file.
showInstrumentationResults fir_filt_circ_buff_typed_entry_point_mex
[y4,z4,p4] = fir_filt_circ_buff_typed_codegen(b4,x4,z4,p4,T4);
8-49
8 Fixed-Point Conversion
• Hover over the "+" in this line of MATLAB code in the instrumented Code
Generation Report:
The reason the sum overflowed is that a full-precision product for b(j)*z(k)
produces a numerictype(true,32,31) because b has numerictype(true,16,17)
and z has numerictype(true,16,14). The sum type is set to "keep least
significant bits" (KeepLSB), so the sum has numerictype(true,32,31).
However, 2 integer bits are necessary to store the minimum and maximum
simulated values of -1.0045 and +1.035, respectively.
T.b=fi([],true,16,16,F);
Run the test file with the new settings and plot the results.
[y1,y2,y3,y4,y5] = fir_filt_circ_buff_typed_test(b,x);
You can see that the overflow has been avoided. However, the plots show a
bias and a larger error due to using C’s natural floor rounding. If this bias is
acceptable to you, then you can stop here and the generated C-code is very
clean.
fir_filt_circ_buff_plot2(6,titles,t,x,y1,y5);
8-50
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
If the bias is not acceptable in your application, then change the rounding
method to ’Nearest’ to eliminate the bias. Rounding to nearest generates
slightly more complicated C-code, but it may be necessary for you if you want
to eliminate the bias and have a smaller error.
The final fixed-point types table with nearest rounding and adjusted
coefficient fraction length is:
function T = fir_filt_circ_buff_dsp_nearest_types()
F = fimath('RoundingMethod','Nearest',...
'OverflowAction','Wrap',...
'ProductMode','KeepLSB',...
'ProductWordLength',32,...
'SumMode','KeepLSB',...
'SumWordLength',32);
T.acc=fi([],true,32,30,F);
T.p=int16([]);
T.b=fi([],true,16,16,F);
T.x=fi([],true,16,14,F);
T.y=fi([],true,16,14,F);
T.z=fi([],true,16,14,F);
end
Call this types table from the entry-point function and run and plot the output.
[y1,y2,y3,y4,y5,y6] = fir_filt_circ_buff_typed_test(b,x);
fir_filt_circ_buff_plot2(7,titles,t,x,y1,y6);
8-51
8 Fixed-Point Conversion
entry-point function or test file so the C-code for the core algorithm can be
included in a larger project.
function fir_filt_circ_buff_build_function()
%
% Declare input arguments
%
T = fir_filt_circ_buff_dsp_nearest_types();
b = zeros(1,12,'like',T.b);
x = zeros(256,1,'like',T.x);
z = zeros(size(b),'like',T.z);
p = cast(0,'like',T.p);
%
% Code generation configuration
%
h = coder.config('lib');
h.PurelyIntegerCode = true;
h.SaturateOnIntegerOverflow = false;
h.SupportNonFinite = false;
h.HardwareImplementation.ProdBitPerShort = 8;
h.HardwareImplementation.ProdBitPerInt = 16;
h.HardwareImplementation.ProdBitPerLong = 32;
%
% Generate C-code
%
codegen fir_filt_circ_buff_typed_codegen -args {b,x,z,p,T} -config h -l
end
Generated C-Code
8-52
Implement FIR Filter Algorithm for Floating-Point and Fixed-Point Types using cast and zeros
(*p)++;
if (*p > 12) {
*p = 1;
}
z[*p - 1] = x[n];
acc = 0L;
k = *p;
for (j = 0; j < 12; j++) {
acc += (int32_T)b[j] * z[k - 1];
k--;
if (k < 1) {
k = 12;
}
}
y[n] = (int16_T)((acc >> 16) + ((acc & 32768L) != 0L));
}
}
fipref(FIPREF_STATE);
clearInstrumentationResults fir_filt_circ_buff_original_entry_point_mex
clearInstrumentationResults fir_filt_circ_buff_typed_entry_point_mex
clear fir_filt_circ_buff_original_entry_point_mex
clear fir_filt_circ_buff_typed_entry_point_mex
tempdirObj.cleanUp;
8-53
8 Fixed-Point Conversion
Prerequisites
• MATLAB
• MATLAB Coder
• Fixed-Point Designer
• C compiler (for most platforms, a default C compiler is
supplied with MATLAB) For a list of supported compilers, see
http://www.mathworks.com/support/compilers/current_release/
You can use mex -setup to change the default compiler. See “Changing
Default Compiler”.
It is best practice to create a separate test script to do all the pre- and
post-processing such as loading inputs, setting up input values, calling the
function under test, and outputting test results. For more information, see
“Create a Test File” on page 8-89.
8-54
Propose Data Types Based on Simulation Ranges
y = zeros(size(x));
for i=1:length(x)
y(i) = b(1)*x(i) + z(1);
z(1) = b(2)*x(i) + z(2) - a(2) * y(i);
z(2) = b(3)*x(i) - a(3) * y(i);
end
end
The test script runs the fun_with_matlab function with three input signals:
chirp, step, and impulse to cover the full intended operating range of the
system. The script then plots the outputs.
% fun_with_matlab_test
%
% Define representative inputs
N = 256; % Number of points
t = linspace(0,1,N); % Time vector from 0 to 1 second
f1 = N/2; % Target frequency of chirp set to Nyquist
x_chirp = sin(pi*f1*t.^2); % Linear chirp from 0 to Fs/2 Hz in 1 second
8-55
8 Fixed-Point Conversion
disp('Test complete.');
The code generation readiness tool screens the code for features and functions
that are not supported for code generation. The tool reports that the
fun_with_matlab.m function is already suitable for code generation.
8-56
Propose Data Types Based on Simulation Ranges
If your entry-point function is not suitable for code generation, the tool
provides a report that lists the source files that contain unsupported features
and functions. The report also provides an indication of how much work
you must do to make the MATLAB code ready for code generation. Before
proposing data types, you must fix these issues. For more information, see
“Detect and Debug Code Generation Errors” on page 9-27.
1 Navigate to the work folder that contains the file for this example.
8-57
8 Fixed-Point Conversion
3 On the project Overview tab, click the Add files link. Browse to the file
fun_with_matlab.m and then click OK to add the file to the project.
The test file runs and displays the outputs of the filter for each of the input
signals.
8-58
Propose Data Types Based on Simulation Ranges
MATLAB Coder determines the input types from the test file and then
displays them.
8-59
8 Fixed-Point Conversion
3 In the Autodefine Input Types dialog box, click Use These Types.
Fixed-Point Conversion
8-60
Propose Data Types Based on Simulation Ranges
The project indicates that you must first define the fixed-point data types.
8-61
8 Fixed-Point Conversion
8-62
Propose Data Types Based on Simulation Ranges
If the MEX function generation fails, the tool provides error message links
to help you navigate to the code that caused the build issues. If your
code contains functions that are not supported for fixed-point conversion,
the tool displays these on the Function Replacements tab. For more
information, see “Running a Simulation”.
8-63
8 Fixed-Point Conversion
The simulation runs and the conversion tool displays a color-coded code
coverage bar to the left of the MATLAB code. Review this information to
verify that the test file is testing the algorithm adequately. Here, the dark
green line to the left of the code indicates that the code is run every time
the algorithm is executed. The orange bar indicates that the code next to
it is executed only once. In this example, this is the expected behavior
because the code is initializing a persistent variable. If your test file is not
covering all your code, update the test or add more test files.
8-64
Propose Data Types Based on Simulation Ranges
If a value has ... next to it, the value is rounded. Place your cursor over
the ... to view the actual value.
8-65
8 Fixed-Point Conversion
6 Examine the proposed types and verify that they cover the full simulation
range. To view logged histogram data for a variable, click its Proposed
Type field.
8-66
Propose Data Types Based on Simulation Ranges
To modify the proposed data types, either enter the required type into the
ProposedType field or use the histogram controls. For more information
about the histogram, see “Histogram”.
7 To validate the build using the proposed types, click Validate Types.
If the errors or warnings occur during validation, they are displayed on the
Type Validation Output tab. For more information, see “Validating
Types”.
8 Click Test Numerics, select Log inputs and outputs for comparison
plots, and then click the Test Numerics button.
8-67
8 Fixed-Point Conversion
The tool runs the test file that you used to define input types to test the
fixed-point MATLAB code. Optionally, you can add test files and select to
run more than one test file to test numerics. The software runs both a
floating-point and a fixed-point simulation and then calculates the errors
for the output variable y. Because you selected to log inputs and outputs
for comparison plots, the tool generates an additional plot for each scalar
output.
The maximum error is less than 0.03%. For the purpose of this example,
this margin of error is acceptable, so you are ready to generate fixed-point
C code.
If the difference is not acceptable, modify the fixed-point data types or your
original algorithm. For more information, see “Testing Numerics”.
8-68
Propose Data Types Based on Simulation Ranges
The code generation report opens and displays the generated code for
fun_with_matlab_fixpt.c. In the generated C code, the variables are not
assigned real types, they are assigned fixed-point data types.
The fimath Product mode and Sum mode settings are both set to
FullPrecision. In FullPrecision mode, the product word length grows
to the sum of the word lengths of the operands.
Because you have changed type proposal settings, you must validate the
types again.
The maximum error is still less than 0.03%, so you are ready to generate
fixed-point C code.
8-69
8 Fixed-Point Conversion
generated code do not exceed 32 bits, the generated code does not contain
utility functions.
i0 = 25593 * x[i];
if (i0 >= 0) {
i2 = (int)((unsigned int)i0 >> 1);
} else {
i2 = ~(int)((unsigned int)~i0 >> 1);
}
8-70
Propose Data Types Based on Simulation Ranges
if (i0 >= 0) {
z[0] = (short)((unsigned int)i0 >> 15);
} else {
z[0] = (short)~(int)((unsigned int)~i0 >> 15);
}
i0 = 25593 * x[i];
if (i0 >= 0) {
i3 = (int)((unsigned int)i0 >> 2);
} else {
i3 = ~(int)((unsigned int)~i0 >> 2);
}
i0 = i3 - 21844 * b_y;
if (i0 >= 0) {
z[1] = (short)((unsigned int)i0 >> 15);
} else {
z[1] = (short)~(int)((unsigned int)~i0 >> 15);
}
y[i] = b_y;
}
}
8-71
8 Fixed-Point Conversion
Prerequisites
• MATLAB
• MATLAB Coder
• Fixed-Point Designer
• C compiler (for most platforms, a default C compiler is
supplied with MATLAB) For a list of supported compilers, see
http://www.mathworks.com/support/compilers/current_release/
You can use mex -setup to change the default compiler. See “Changing
Default Compiler”.
3 Copy the dti.m and dti_test.m files to your local working folder.
It is best practice to create a separate test script to do all the pre- and
post-processing such as loading inputs, setting up input values, calling the
function under test, and outputting test results. For more information, see
“Create a Test File” on page 8-89.
8-72
Propose Data Types Based on Derived Ranges
% Compute Output
if (u_state > limit_upper)
y = limit_upper;
clip_status = -2;
elseif (u_state >= limit_upper)
y = limit_upper;
clip_status = -1;
elseif (u_state < limit_lower)
y = limit_lower;
clip_status = 2;
elseif (u_state <= limit_lower)
8-73
8 Fixed-Point Conversion
y = limit_lower;
clip_status = 1;
else
y = u_state;
clip_status = 0;
end
% Update State
tprod = gain_val * u_in;
u_state = y + tprod;
function b = subFunction(a)
b = a*a;
The test script runs the dti function with a sine wave input. The script then
plots the input and output signals.
% dti_test
% cleanup
clear dti
% input signal
x_in = sin(2.*pi.*(0:0.001:2)).';
pause(10);
len = length(x_in);
y_out = zeros(1,len);
is_clipped_out = zeros(1,len);
for ii=1:len
data = x_in(ii);
% call to the dti function
init_val = 0;
gain_val = 1;
upper_limit = 500;
lower_limit = -500;
8-74
Propose Data Types Based on Derived Ranges
end
subplot(2,1,2); plot(1:len,y_out);
xlabel('Time')
ylabel('Amplitude')
title('Output Signal (DTI)')
disp('Test complete.');
In the current working folder, right-click the dti.m function. From the context
menu, select Check Code Generation Readiness.
The code generation readiness tool screens the code for features and functions
that are not supported for code generation. The tool reports that the dti.m
function is already suitable for code generation.
8-75
8 Fixed-Point Conversion
If your entry-point function is not suitable for code generation, the tool
provides a report that lists the source files that contain unsupported features
and functions. The report also provides an indication of how much work
you must do to make the MATLAB code ready for code generation. Before
proposing data types, you must fix these issues. For more information, see
“Detect and Debug Code Generation Errors” on page 9-27.
1 Navigate to the work folder that contains the file for this example.
2 On the MATLAB Apps tab, select MATLAB Coder and then, in the
MATLAB Coder Project dialog box, set Name to dti.prj.
3 On the project Overview tab, click the Add files link. Browse to the file
dti.m and then click OK to add the file to the project.
8-76
Propose Data Types Based on Derived Ranges
2 In the Autodefine Input Types dialog box, add dti_test as a test file and
then click Run.
The test file runs and displays the outputs of the filter for each of the input
signals.
8-77
8 Fixed-Point Conversion
MATLAB Coder determines the input types from the test file and then
displays them.
3 In the Autodefine Input Types dialog box, click Use These Types.
Fixed-Point Conversion
The project indicates that you must first define the fixed-point data types.
8-78
Propose Data Types Based on Derived Ranges
8-79
8 Fixed-Point Conversion
If the MEX function generation fails, the tool provides error message links
to help you navigate to the code that caused the build issues. If your
code contains functions that are not supported for fixed-point conversion,
the tool displays these on the Function Replacements tab. For more
information, see “Running a Simulation”.
8-80
Propose Data Types Based on Derived Ranges
Range analysis computes the derived ranges and displays them in the
Variables tab. Using these derived ranges, the analysis proposes
fixed-point types for each variable based on the default type proposal
settings, and displays them in the Proposed Type column. The Validate
Types option is now enabled.
% Compute Output
if (u_state > limit_upper)
y = limit_upper;
clip_status = -2;
elseif (u_state >= limit_upper)
y = limit_upper;
clip_status = -1;
elseif (u_state < limit_lower)
y = limit_lower;
clip_status = 2;
elseif (u_state <= limit_lower)
y = limit_lower;
clip_status = 1;
else
y = u_state;
clip_status = 0;
end
8-81
8 Fixed-Point Conversion
The tool provides a Quick derived range analysis option and the option
to specify a timeout in case the analysis takes a very long time. For more
information, see “Computing Derived Ranges”
8-82
Propose Data Types Based on Derived Ranges
5 To validate the build using the proposed types, click Validate Types.
If the errors or warnings occur during validation, they are displayed on the
Type Validation Output tab. For more information, see “Validating
Types”.
6 Run the test file to test the fixed-point MATLAB code. Click Test
Numerics and select Log inputs and outputs for comparison plots,
and then click the Test Numerics button.
8-83
8 Fixed-Point Conversion
The tool runs the test file that you used to define input types to test the
fixed-point MATLAB code. Optionally, you can add test files and select to
run more than one test file to test numerics. The software runs both a
floating-point and a fixed-point simulation and then calculates the errors
for the output variables y and clip_status. Because you selected to log
inputs and outputs for comparison plots, the tool generates an additional
plot for each scalar output.
8-84
Propose Data Types Based on Derived Ranges
8-85
8 Fixed-Point Conversion
If the difference is not acceptable, modify the fixed-point data types or your
original algorithm. For more information, see “Testing Numerics”.
MATLAB Coder builds the project and generates a C static library and
supporting files in the default subfolder, codegen/lib/dti_FixPt.
The code generation report opens and displays the generated code for
dti_FixPt.c. In the generated C code, variables are assigned fixed-point
data types.
8-86
Manual Fixed-Point Conversion Workflow
2 Write a test file that calls your original MATLAB algorithm to validate the
behavior of your algorithm.
6 Run your original MATLAB algorithm to log min/max data. View this
data in the instrumentation report.
8-87
8 Fixed-Point Conversion
Create fixed-point types table based on proposed data types, build mex,
run, and then compare against baseline.
8-88
Manual Fixed-Point Conversion Best Practices
8-89
8 Fixed-Point Conversion
Your test files should exercise the algorithm over its full operating range so
that the simulation ranges are accurate. For example, for a filter, realistic
inputs are impulses, sums of sinusoids, and chirp signals. With these inputs,
8-90
Manual Fixed-Point Conversion Best Practices
using linear theory, you can verify that the outputs are correct. Signals that
produce maximum output are useful for verifying that your system does not
overflow. The quality of the proposed fixed-point data types depends on how
well the test files cover the operating range of the algorithm with the accuracy
that you want.
• Instrument your code and provide data type proposals to help you convert
your algorithm to fixed point, using the following functions:
- buildInstrumentedMex, which generates compiled C code that includes
logging instrumentation.
- showInstrumentationResults, which shows the results logged by the
instrumented, compiled C code.
- clearInstrumentationResults, which clears the logged
instrumentation results from memory.
• Accelerate your fixed-point algorithms by creating a MEX file using the
fiaccel function.
• Add the %#codegen pragma to the top of your MATLAB file. The MATLAB
code analyzer flags functions and constructs that are not available in the
subset of the MATLAB language supported for code generation. This advice
appears in real-time as you edit your code in the MATLAB editor.
8-91
8 Fixed-Point Conversion
For more information, see “Check Code Using the MATLAB Code Analyzer”
on page 9-122.
• Use the Code Generation Readiness tool to generate a static report
on your code. The report identifies calls to functions and the use of
data types that are not supported for code generation. To generate
a report for a function, myFunction1, at the command line, enter
coder.screener('myFunction1').
For more information, see “Check Code Using the Code Generation
Readiness Tool” on page 9-121.
8-92
Manual Fixed-Point Conversion Best Practices
8-93
8 Fixed-Point Conversion
2 Create a table of data type definitions, starting with original data types
used in your code. Before converting to fixed point, create a data type table
that uses all single data types to find type mismatches and other problems.
3 Run your code connected to each table and look at the results to verify
the connection.
8-94
Manual Fixed-Point Conversion Best Practices
Separating data type specifications from algorithm code enables you to:
8-95
8 Fixed-Point Conversion
% Run
[y,z] = myfilter_mex(b,x,z);
% Show
showInstrumentationResults myfilter_mex ...
8-96
Manual Fixed-Point Conversion Best Practices
Because the data types, not the values, are used, specify the prototype values
as empty ([]).
In some cases, it might be more efficient to leave some parts of the code in
floating point. For example, when there is high dynamic range or that part of
the code is sensitive to round-off errors.
8-97
8 Fixed-Point Conversion
Algorithm Code
function [y,z]=myfilter(b,x,z,T)
y = zeros(size(x),'like',T.y);
for n=1:length(x)
z(:) = [x(n); z(1:end-1)];
y(n) = b * z;
end
end
Types Tables
function T = mytypes(dt)
switch dt
case 'double'
T.b = double([]);
T.x = double([]);
T.y = double([]);
case 'fixed16'
T.b = fi([],true,16,15);
T.x = fi([],true,16,15);
T.y = fi([],true,16,14);
end
end
8-98
Manual Fixed-Point Conversion Best Practices
Test File
% Test inputs
b = fir1(11,0.25);
t = linspace(0,10*pi,256)';
x = sin((pi/16)*t.^2);
% Linear chirp
% Cast inputs
T=mytypes('fixed16');
b=cast(b,'like',T.b);
x=cast(x,'like',T.x);
z=zeros(size(b'),'like',T.x);
% Run
[y,z] = myfilter(b,x,z,T);
8-99
8 Fixed-Point Conversion
a =
3.1416
a =
3.1416
Overflows indicated in red in the Code Generation Report show in the "outside
range" bin in the NumericTypeScope. Launch the NumericTypeScope for an
associated variable or expression by clicking on the histogram view icon .
8-100
Manual Fixed-Point Conversion Best Practices
Algorithm Code
Types Tables
function T = mytypes(dt)
switch dt
case 'double'
T.b = double([]);
T.x = double([]);
T.y = double([]);
case 'fixed8'
T.b = fi([],true,8,7);
T.x = fi([],true,8,7);
T.y = fi([],true,8,6);
case 'fixed16'
T.b = fi([],true,16,15);
T.x = fi([],true,16,15);
T.y = fi([],true,16,14);
end
end
8-101
8 Fixed-Point Conversion
Test File
function mytest
% Test inputs
b = fir1(11,0.25);
t = linspace(0,10*pi,256)';
x = sin((pi/16)*t.^2); % Linear chirp
% Run
y0 = entrypoint('double',b,x);
y8 = entrypoint('fixed8',b,x);
y16 = entrypoint('fixed16',b,x);
% Plot
subplot(3,1,1);plot(t,x,'c',t,y0,'k');
legend('Input','Baseline output')
title('Baseline')
subplot(3,2,3);plot(t,y8,'k');
title('8-bit fixed-point output')
subplot(3,2,4);plot(t,y0-double(y8),'r');
title('8-bit fixed-point error')
subplot(3,2,5);plot(t,y16,'k');
title('16-bit fixed-point output')
xlabel('Time (s)')
subplot(3,2,6);plot(t,y0-double(y16),'r');
title('16-bit fixed-point error')
xlabel('Time (s)')
end
8-102
Manual Fixed-Point Conversion Best Practices
8-103
8 Fixed-Point Conversion
i0 = i2 + i3;
if ((i0 & 65536) != 0) {
y = i0 | -65536;
} else {
y = i0 & 65535;
}
return y;
}
F = fimath(...
'RoundingMethod','Floor', ...
'OverflowAction','Wrap', ...
'ProductMode','KeepLSB', ...
'ProductWordLength',32, ...
'SumMode','KeepLSB', ...
'SumWordLength',32);
T.a = fi([],1,16,0,F);
T.b = fi([],1,16,0,F);
a = cast(0,'like',T.a);
b = cast(0,'like',T.b);
8-104
Manual Fixed-Point Conversion Best Practices
• Use bit shifting when the denominator is a power of two. For example,
bitsra(x,3) instead of x/8.
• Multiply by the inverse when the denominator is constant. For example,
x*0.2 instead of x/5.
8-105
8 Fixed-Point Conversion
8-106
Fixed-Point Code for MATLAB® Classes
Unsupported Constructs
The automated conversion process does not support:
• Class inheritance.
8-107
8 Fixed-Point Conversion
• Packages.
• Constructors that use nargin and varargin.
properties(Constant)
MAX_VALUE = 128
end
methods
function out = next(this)
out = this.Count;
if this.Value == this.MAX_VALUE
this.Value = 0;
else
this.Value = this.Value + 1;
end
end
end
end
8-108
Fixed-Point Code for MATLAB® Classes
end
methods(Static)
function t = MAX_VALUE()
t = 128;
end
end
methods
function this = Counter()
this.Value = 0;
end
function out = next(this)
out = this.Value;
if this.Value == this.MAX_VALUE
this.Value = 0;
else
this.Value = this.Value + 1;
end
end
end
end
8-109
8 Fixed-Point Conversion
8-110
Automated Fixed-Point Conversion Best Practices
8-111
8 Fixed-Point Conversion
that the simulation ranges are accurate. For example, for a filter, realistic
inputs are impulses, sums of sinusoids, and chirp signals. With these inputs,
using linear theory, you can verify that the outputs are correct. Signals that
produce maximum output are useful for verifying that your system does not
overflow. The quality of the proposed fixed-point data types depends on how
well the test files cover the operating range of the algorithm with the accuracy
that you want. Reviewing code coverage results help you verify that your test
file is exercising the algorithm adequately. Review code flagged with a red
code coverage bar because this code is not executed. If the code coverage is
inadequate, modify the test file or add more test files to increase coverage.
For more information see “Code Coverage” on page 8-6.
8-112
Automated Fixed-Point Conversion Best Practices
8-113
8 Fixed-Point Conversion
fixed-point properties such as the available word length, fraction length, and
math modes, as well as available math libraries.
For more information, see “Specify Type Proposal Options” on page 8-20.
• To view the histogram for a variable, on the Variables tab, click the
Proposed Type field for that variable.
8-114
Automated Fixed-Point Conversion Best Practices
You can view the effect of changing the proposed data types by dragging the
edges of the bounding box in the histogram window to change the proposed
data type and selecting or clearing the Signed option.
• If the values overflow and the range cannot fit the proposed type, the table
shows proposed types in red.
When the tool applies data types, it generates an html report that provides
overflow information and highlights overflows in red. Review the proposed
data types.
8-115
8 Fixed-Point Conversion
C. The KeepLSB setting for ProductMode and SumMode models the behavior of
integer operations in the C language, while KeepMSB models the behavior of
many DSP devices. Different rounding methods require different amounts
of overhead code. Setting the RoundingMethod property to Floor, which
is equivalent to two’s complement truncation, provides the most efficient
rounding implementation. Similarly, the standard method for handling
overflows is to wrap using modulo arithmetic. Other overflow handling
methods create costly logic. Whenever possible, set OverflowAction to Wrap.
8-116
Automated Fixed-Point Conversion Best Practices
i0 = i2 + i3;
if ((i0 & 65536) != 0) {
y = i0 | -65536;
} else {
y = i0 & 65535;
}
return y;
}
Fix
int adder(short a, short b)
To make the generated C code more {
efficient, choose fixed-point math return a + b;
settings that match your processor }
types.
To customize fixed-point type
proposals, use the Fixed-Point
Conversion tool Advanced settings.
Select fimath and then set:
Rounding method Floor
Overflow action Wrap
Product mode KeepLSB
Sum mode KeepLSB
Product word 32
length
Sum word length 32
8-117
8 Fixed-Point Conversion
• Use bit shifting when the denominator is a power of two. For example,
bitsra(x,3) instead of x/8.
• Multiply by the inverse when the denominator is constant. For example,
x*0.2 instead of x/5.
• If the divisor is not constant, use a temporary variable for the division.
Doing so results in a more efficient data type proposal and, if overflows
occur, makes it easier to see which expression is overflowing.
8-118
Automated Fixed-Point Conversion Best Practices
8-119
8 Fixed-Point Conversion
8-120
9
9-2
Code Acceleration and Code Generation from MATLAB®
9-3
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-4
Functions Supported for Code Acceleration or C Code Generation
Function Remarks/Limitations
abs N/A
accumneg N/A
accumpos N/A
add • Code generation in MATLAB does not support the syntax
F.add(a,b). You must use the syntax add(F,a,b).
all N/A
any N/A
9-5
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Function Remarks/Limitations
atan2 N/A
bitand Not supported for slope-bias scaled fi objects.
bitandreduce N/A
bitcmp N/A
bitconcat N/A
bitget N/A
bitor Not supported for slope-bias scaled fi objects.
bitorreduce N/A
bitreplicate N/A
bitrol N/A
bitror N/A
bitset N/A
bitshift N/A
bitsliceget N/A
bitsll Generated code may not handle out of range shifting.
bitsra Generated code may not handle out of range shifting.
bitsrl Generated code may not handle out of range shifting.
bitxor Not supported for slope-bias scaled fi objects.
bitxorreduce N/A
ceil N/A
complex N/A
conj N/A
9-6
Functions Supported for Code Acceleration or C Code Generation
Function Remarks/Limitations
conv • Variable-sized inputs are only supported when the SumMode
property of the governing fimath is set to Specify precision or
Keep LSB.
• For variable-sized signals, you may see different results between
generated code and MATLAB.
- In the generated code, the output for variable-sized signals is
computed using the SumMode property of the governing fimath.
- In MATLAB, the output for variable-sized signals is computed
using the SumMode property of the governing fimath when both
inputs are nonscalar. However, if either input is a scalar,
MATLAB computes the output using the ProductMode of the
governing fimath.
convergent N/A
cordicabs Variable-size signals are not supported.
cordicangle Variable-size signals are not supported.
cordicatan2 Variable-size signals are not supported.
cordiccart2pol Variable-size signals are not supported.
cordiccexp Variable-size signals are not supported.
cordiccos Variable-size signals are not supported.
cordicpol2cart Variable-size signals are not supported.
cordicrotate Variable-size signals are not supported.
cordicsin Variable-size signals are not supported.
cordicsincos Variable-size signals are not supported.
cos N/A
ctranspose N/A
diag If supplied, the index, k, must be a real and scalar integer value that
is not a fi object.
9-7
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Function Remarks/Limitations
divide • Any non-fi input must be constant; that is, its value must be
known at compile time so that it can be cast to a fi object.
• Complex and imaginary divisors are not supported.
• Code generation in MATLAB does not support the syntax
T.divide(a,b).
double N/A
end N/A
eps • Supported for scalar fixed-point signals only.
• Supported for scalar, vector, and matrix, fi single and fi double
signals.
eq Not supported for fixed-point signals with different biases.
fi • The default constructor syntax without any input arguments is
not supported.
• If the numerictype is not fully specified, the input to fi must be
a constant, a fi, a single, or a built-in integer value. If the input
is a built-in double value, it must be a constant. This limitation
allows fi to autoscale its fraction length based on the known data
type of the input.
• All properties related to data type must be constant for code
generation.
• numerictype object information must be available for
nonfixed-point Simulink inputs.
filter • Variable-sized inputs are only supported when the SumMode
property of the governing fimath is set to Specify precision or
Keep LSB.
9-8
Functions Supported for Code Acceleration or C Code Generation
Function Remarks/Limitations
fimath • Fixed-point signals coming in to a MATLAB Function block from
Simulink are assigned a fimath object. You define this object in
the MATLAB Function block dialog in the Model Explorer.
• Use to create fimath objects in the generated code.
• If the ProductMode property of the fimath object is set to
anything other than FullPrecision, the ProductWordLength and
ProductFractionLength properties must be constant.
• If the SumMode property of the fimath object is set to
anything other than FullPrecision, the SumWordLength and
SumFractionLength properties must be constant.
fix N/A
fixed.Quantizer N/A
floor N/A
ge Not supported for fixed-point signals with different biases.
get The syntax structure = get(o) is not supported.
getlsb N/A
getmsb N/A
gt Not supported for fixed-point signals with different biases.
hdlram N/A
horzcat N/A
imag N/A
int8, int16, int32, N/A
int64
iscolumn N/A
isempty N/A
isequal N/A
isfi N/A
isfimath N/A
9-9
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Function Remarks/Limitations
isfimathlocal N/A
isfinite N/A
isinf N/A
isnan N/A
isnumeric N/A
isnumerictype N/A
isreal N/A
isrow N/A
isscalar N/A
issigned N/A
isvector N/A
le Not supported for fixed-point signals with different biases.
length N/A
logical N/A
lowerbound N/A
lsb • Supported for scalar fixed-point signals only.
• Supported for scalar, vector, and matrix, fi single and double
signals.
lt Not supported for fixed-point signals with different biases.
max N/A
mean N/A
median N/A
min N/A
minus Any non-fi input must be constant; that is, its value must be known
at compile time so that it can be cast to a fi object.
9-10
Functions Supported for Code Acceleration or C Code Generation
Function Remarks/Limitations
mpower • When the exponent k is a variable and the input is a scalar,
the ProductMode property of the governing fimath must be
SpecifyPrecision.
• When the exponent k is a variable and the input is not
scalar, the SumMode property of the governing fimath must be
SpecifyPrecision.
• Variable-sized inputs are only supported when the SumMode
property of the governing fimath is set to SpecifyPrecision or
Keep LSB.
• For variable-sized signals, you may see different results between
the generated code and MATLAB.
- In the generated code, the output for variable-sized signals is
computed using the SumMode property of the governing fimath.
- In MATLAB, the output for variable-sized signals is computed
using the SumMode property of the governing fimath when
the first input, a, is nonscalar. However, when a is a scalar,
MATLAB computes the output using the ProductMode of the
governing fimath.
mpy • Code generation in MATLAB does not support the syntax
F.mpy(a,b). You must use the syntax mpy(F,a,b).
• When you provide complex inputs to the mpy function inside of a
MATLAB Function block, you must declare the input as complex
before running the simulation. To do so, go to the Ports and
data manager and set the Complexity parameter for all known
complex inputs to On.
mrdivide N/A
9-11
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Function Remarks/Limitations
mtimes • Any non-fi input must be constant; that is, its value must be
known at compile time so that it can be cast to a fi object.
• Variable-sized inputs are only supported when the SumMode
property of the governing fimath is set to SpecifyPrecision or
KeepLSB.
• For variable-sized signals, you may see different results between
the generated code and MATLAB.
- In the generated code, the output for variable-sized signals is
computed using the SumMode property of the governing fimath.
- In MATLAB, the output for variable-sized signals is computed
using the SumMode property of the governing fimath when both
inputs are nonscalar. However, if either input is a scalar,
MATLAB computes the output using the ProductMode of the
governing fimath.
ndims N/A
ne Not supported for fixed-point signals with different biases.
nearest N/A
numberofelements numberofelements will be removed in a future release. Use numel
instead.
numel N/A
numerictype • Fixed-point signals coming in to a MATLAB Function block from
Simulink are assigned a numerictype object that is populated
with the signal’s data type and scaling information.
• Returns the data type when the input is a nonfixed-point signal.
• Use to create numerictype objects in generated code.
• All numerictype object properties related to the data type must
be constant.
permute N/A
plus Any non-fi inputs must be constant; that is, its value must be known
at compile time so that it can be cast to a fi object.
9-12
Functions Supported for Code Acceleration or C Code Generation
Function Remarks/Limitations
pow2 N/A
power When the exponent k is a variable, the ProductMode property of the
governing fimath must be SpecifyPrecision.
qr N/A
quantize N/A
range N/A
rdivide N/A
real N/A
realmax N/A
realmin N/A
reinterpretcast N/A
removefimath N/A
repmat N/A
rescale N/A
reshape N/A
round N/A
setfimath N/A
sfi • All properties related to data type must be constant for code
generation.
sign N/A
sin N/A
single N/A
size N/A
sort N/A
sqrt • Complex and [Slope Bias] inputs error out.
• Negative inputs yield a 0 result.
storedInteger N/A
9-13
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Function Remarks/Limitations
storedIntegerToDouble N/A
sub • Code generation in MATLAB does not support the syntax
F.sub(a,b). You must use the syntax sub(F,a,b).
subsasgn N/A
subsref N/A
sum Variable-sized inputs are only supported when the SumMode property
of the governing fimath is set to Specify precision or Keep LSB.
times • Any non-fi input must be constant; that is, its value must be
known at compile time so that it can be cast to a fi object.
• When you provide complex inputs to the times function inside of a
MATLAB Function block, you must declare the input as complex
before running the simulation. To do so, go to the Ports and
data manager and set the Complexity parameter for all known
complex inputs to On.
transpose N/A
tril If supplied, the index, k, must be a real and scalar integer value that
is not a fi object.
triu If supplied, the index, k, must be a real and scalar integer value that
is not a fi object.
ufi • All properties related to data type must be constant for code
generation.
uint8, uint16, uint32, N/A
uint64
uminus N/A
uplus N/A
upperbound N/A
vertcat N/A
9-14
Workflow for Fixed-Point Code Acceleration and Generation
9-15
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Set Up C Compiler
Fixed-Point Designer automatically locates and uses a supported installed
compiler. For the current list of supported compilers, see Supported and
Compatible Compilers on the MathWorks® Web site.
You can use mex -setup to change the default compiler. See “Changing
Default Compiler”.
9-16
Accelerate Code Using fiaccel
Running fiaccel
The basic command is:
fiaccel M_fcn
• Searches for the function M_fcn stored in the file M_fcn.m as specified in
“Compile Path Search Order” on page 9-24.
• Compiles M_fcn to MEX code.
• If there are no errors or warnings, generates a platform-specific MEX file in
the current folder, using the naming conventions described in “File Naming
Conventions” on page 9-41.
• If there are errors, does not generate a MEX file, but produces an error
report in a default output folder, as described in “Generated Files and
Locations” on page 9-18.
9-17
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
You can modify this default behavior by specifying one or more compiler
options with fiaccel, separated by spaces on the command line.
Generates: In:
Platform-specific MEX files Current folder
HTML reports Default output folder:
(if errors or warnings occur during fiaccel/mex/M_fcn_name/html
compilation)
You can change the name and location of generated files by using the options
-o and -d when you run fiaccel.
In this example, you will use the fiaccel function to compile different parts
of a simple algorithm. By comparing the run times of the two cases, you will
see the benefits and best use of the fiaccel function.
function B = fi_matrix_column_sum(A)
% Sum the columns of matrix A.
%#codegen
[m,n] = size(A);
w = get(A,'WordLength') + ceil(log2(m));
f = get(A,'FractionLength');
B = fi(zeros(1,n),true,w,f);
9-18
Accelerate Code Using fiaccel
for j = 1:n
for i = 1:m
B(j) = B(j) + A(i,j);
end
end
The first portion of code executes the algorithm using only MATLAB
functions. The second portion of the code compiles the entire algorithm using
the fiaccel function. The MATLAB tic and toc functions keep track of the
run times for each method of execution.
% MATLAB
fipref('NumericTypeDisplay','short');
A = fi(randn(1000,10));
tic
B = fi_matrix_column_sum(A)
t_matrix_column_sum_m = toc
% fiaccel
fiaccel fi_matrix_column_sum -args {A} ...
-I [matlabroot '/toolbox/fixedpoint/fidemos']
tic
B = fi_matrix_column_sum_mex(A);
t_matrix_column_sum_mex = toc
9-19
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
The first portion of code executes the algorithm using only MATLAB functions.
The second portion of the code compiles the smallest unit of computation with
the fiaccel function, leaving the rest of the computations to MATLAB.
% MATLAB
tic
[m,n] = size(A);
w = get(A,'WordLength') + ceil(log2(m));
f = get(A,'FractionLength');
B = fi(zeros(1,n),true,w,f);
for j = 1:n
for i = 1:m
B(j) = fi_scalar_sum(B(j),A(i,j));
% B(j) = B(j) + A(i,j);
end
end
t_scalar_sum_m = toc
% fiaccel
fiaccel fi_scalar_sum -args {B(1),A(1,1)} ...
-I [matlabroot '/toolbox/fixedpoint/fidemos']
tic
[m,n] = size(A);
w = get(A,'WordLength') + ceil(log2(m));
f = get(A,'FractionLength');
B = fi(zeros(1,n),true,w,f);
for j = 1:n
for i = 1:m
B(j) = fi_scalar_sum_mex(B(j),A(i,j));
% B(j) = B(j) + A(i,j);
end
end
t_scalar_sum_mex = toc
Ratio of Times
A comparison of Trial 1 and Trial 2 appears in the following table. Your
computer may record different times than the ones the table shows, but the
ratios should be approximately the same. There is an extreme difference
in ratios between the trial where the entire algorithm was compiled using
9-20
Accelerate Code Using fiaccel
You can specify data type override in this example by typing an extra
command at the MATLAB prompt in the “Define Fixed-Point Parameters”
section of the example. To turn data type override on, type the following
command at the MATLAB prompt after running the reset(fipref)
command in that section:
fipref('DataTypeOverride','TrueDoubles')
9-21
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
RoundingMethod: Nearest
OverflowAction: Saturate
ProductMode: FullPrecision
SumMode: FullPrecision
When running MEX functions that depend on the MATLAB default fimath
value, do not change this value during your MATLAB session. Otherwise,
MATLAB generates a warning, alerting you to a mismatch between the
compile-time and run-time fimath values. For example, create the following
MATLAB function:
Generate the MEX function test_mex to use the factory setting of the
MATLAB default fimath.
resetglobalfimath;
fiaccel test
Run test_mex.
test_mex
ans =
0
DataTypeMode: Fixed-point: binary point scaling
Signedness: Signed
WordLength: 16
FractionLength: 15
9-22
Accelerate Code Using fiaccel
Modify the MATLAB default fimath value so it no longer matches the setting
used at compile time.
F = fimath('RoundingMethod','Floor');
globalfimath(F);
clear test_mex
test_mex
testglobalfimath_mex
Warning: This function was generated with a different default fimath than t
ans =
0
DataTypeMode: Fixed-point: binary point scaling
Signedness: Signed
WordLength: 16
FractionLength: 15
To avoid this issue, separate the fimath properties from your algorithm
by using types tables. For more information, see “Separate Data Type
Definitions from Algorithm” on page 8-94.
9-23
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Naming Conventions
MATLAB enforces naming conventions for functions and generated files.
Reserved Prefixes
MATLAB reserves the prefix eml for global C functions and variables in
generated code. For example, run-time library function names all begin with
the prefix emlrt, such as emlrtCallMATLAB. To avoid naming conflicts, do not
name C functions or primary MATLAB functions with the prefix eml.
Reserved Keywords
9-24
File Infrastructure and Paths Setup
MATLAB Coder software reserves certain words for its own use as keywords
of the generated code language. MATLAB Coder keywords are reserved for
use internal to MATLAB Coder software and should not be used in MATLAB
code as identifiers or function names. C reserved keywords should also not be
used in MATLAB code as identifiers or function names. If your MATLAB code
contains any reserved keywords, the code generation build does not complete
and an error message is displayed. To address this error, modify your code
to use identifiers or names that are not reserved.
If you are generating C++ code using the MATLAB Coder software, in
addition, your MATLAB code must not contain the “C++ Reserved Keywords”
on page 9-25.
C Reserved Keywords.
9-25
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-26
Detect and Debug Code Generation Errors
Debugging Strategies
To prepare your algorithms for code generation, MathWorks recommends
that you choose a debugging strategy for detecting and correcting violations
in your MATLAB applications, especially if they consist of a large number of
MATLAB files that call each other’s functions. Here are two best practices:
9-27
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
a. Remove extrinsic
declarations one by one
fiaccel checks for all potential syntax violations at compile time. When
fiaccel detects errors or warnings, it automatically produces a code
9-28
Detect and Debug Code Generation Errors
generation report that describes the issues and provides links to the offending
code. See “Use Fixed-Point Code Generation Reports” on page 9-42.
If your MATLAB code calls functions on the MATLAB path, fiaccel attempts
to compile these functions unless you declare them to be extrinsic.
9-29
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
comp_cfg = coder.mexconfig
MATLAB displays the list of compiler options and their current values in
the command window.
3 Invoke fiaccel with the -config option and specify the configuration
object as its argument:
configuration_object.property = value
9-30
Set Up C Code Compilation Options
co_cfg = coder.mexconfig
co_cfg.InlineThreshold = 25;
co_cfg.InlineStackLimit = 4096;
fiaccel -config co_cfg myFun
9-31
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Debugging
Echo expressions without EchoExpressions Specify whether or not actions
semicolons true, false that do not terminate with
a semicolon appear in the
MATLAB Command Window.
9-32
MEX Configuration Dialog Box Options
9-33
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-34
MEX Configuration Dialog Box Options
9-35
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-36
MEX Configuration Dialog Box Options
See Also
• “Control Run-Time Checks” on page 9-62
• “Variable-Size Data Definition for Code Generation” on page 22-3
• “Generate C Code from Code Containing Global Data” on page 9-48
9-37
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Note Alternatively, you can use the assert function to define properties
of primary function inputs directly in your MATLAB file. For more
information, see “Define Input Properties Programmatically in MATLAB
File” on page 9-54.
9-38
Best Practices for Accelerating Fixed-Point Code
is suitable for code generation. If you do not specify the -report option,
fiaccel generates a report only if build errors or warnings occur.
The code generation report contains the following information:
- Summary of code generation results, including type of target and
number of warnings or errors
- Target build log that records build and linking activities
- Links to generated files
- Error and warning messages (if any)
Build Scripts
Use build scripts to call fiaccel to generate MEX functions from your
MATLAB function.
close all;
clear all;
clc;
N = 73113;
• close all deletes all figures whose handles are not hidden. See close in
the MATLAB Graphics function reference for more information.
9-39
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
• clear all removes all variables, functions, and MEX-files from memory,
leaving the workspace empty. This command also clears all breakpoints.
Note Remove the clear all command from the build scripts if you want
to preserve breakpoints for debugging.
• clc clears all input and output from the Command Window display, giving
you a “clean screen.”
• N = 73113 sets the value of the variable N, which represents the number of
samples in each of the two input parameters for the function lms_02
• fiaccel -report lms_02.m -args { zeros(N,1) zeros(N,1) } calls
fiaccel to accelerate simulation of the file lms_02.m using the following
options:
- -report generates a code generation report
- -args { zeros(N,1) zeros(N,1) } specifies the properties of the
function inputs as a cell array of example values. In this case, the input
parameters are N-by-1 vectors of real doubles.
1 From the MATLAB menu, select File > Preferences > Code Analyzer.
2 Select the Enable integrated warning and error messages check box.
9-40
Best Practices for Accelerating Fixed-Point Code
For example, the file naming convention in the Generating MEX Functions
getting started tutorial is:
For example:
• The file build_01.m is the first version of the build script for this tutorial.
• The file test_03.m is the third version of the test script for this tutorial.
9-41
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-42
Use Fixed-Point Code Generation Reports
avg = x;
for k = 1:length(x)
[avg(k),z] = moving_average(x(k),z);
end
end
3 Use fiaccel to create a MEX function and accelerate the code. Specify
the type of input x using the -args option. Specify the -report option
to create a code generation report.
x = fi(rand(100,1),1,16,15);
fiaccel -report test_moving_average -args {x}
Click the View report link to view the error report and debug your code.
For more information on working with error reports, see “Code Generation
Reports”.
Color Meaning
Green Data type information is available for the
selected variable at this location in the code.
Orange There is a warning message associated with
the selected variable or expression.
9-43
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Color Meaning
Pink No data type information is available for the
selected variable.
Purple Data type information is available for the
selected expression at this location in the
code.
Red There is an error message associated with the
selected variable or expression.
Variables in your code that have data type information available appear
highlighted in green.
9-44
Use Fixed-Point Code Generation Reports
Expressions in your code that have data type information available appear
highlighted in purple, as the next figure shows.
9-45
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
The following figure shows the information displayed for a fi object that is
using default fimath values.
9-46
Use Fixed-Point Code Generation Reports
You can sort the variables by clicking the column headings in the Variables
tab. To sort the variables by multiple columns, press the Shift key while
clicking the column headings.
See Also
For more information about using the code generation report with the
fiaccel function, see the fiaccel reference page.
For information about local and default fimath, see “fimath Object
Construction” on page 4-2.
For information about using the code generation report with the codegen
function, see “Code Generation Reports”.
9-47
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Workflow Overview
To generate MEX functions from MATLAB code that uses global data:
If you use global data, you must also specify whether you want to synchronize
this data between MATLAB and the generated code. If there is no
interaction between MATLAB and the generated code, it is safe to disable
synchronization. Otherwise, you should enable synchronization. For more
information, see “Synchronizing Global Data with MATLAB” on page 9-50.
function y = use_globals()
%#codegen
% Turn off inlining to make
% generated code easier to read
coder.inline('never');
9-48
Generate C Code from Code Containing Global Data
global AR B;
AR = fi(ones(4),1,16,14);
B = fi([1 2 3],1,16,13);
Alternatively, specify the type and initial value with the -globals flag using
the format -globals {'g', {type, initial_value}}.
9-49
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-50
Generate C Code from Code Containing Global Data
9-51
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
You control the synchronization of global data with extrinsic functions using
the coder.extrinsic -sync:on and -sync:off options.
comp_cfg = coder.mexconfig
comp_cfg.GlobalDataSyncMethod = 'SyncAtEntryAndExits';
• Synchronized before and after each extrinsic call if the global data
synchronization mode is At MEX-function entry, exit and extrinsic
calls. If you are sure that certain extrinsic calls do not affect global
data, turn off synchronization for these calls using the -sync:off option.
Turning off synchronization improves performance. For example, if
functions foo1 and foo2 do not affect global data, turn off synchronization
for these functions:
9-52
Generate C Code from Code Containing Global Data
-sync:on option. For example, if functions foo1 and foo2 do affect global
data, turn on synchronization for these functions:
9-53
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Use the assert function to invoke standard MATLAB functions for specifying
the class, size, and complexity of primary function inputs.
Sets the input parameter param to the MATLAB class class_name. For
example, to set the class of input U to a 32-bit signed integer, call:
...
assert(isa(U,'embedded.fi'));
...
9-54
Define Input Properties Programmatically in MATLAB® File
Note If you set the class of an input parameter to fi, you must also set its
numerictype, see “Specify numerictype of Fixed-Point Input” on page 9-57.
You can also set its fimath properties, see “Specify fimath of Fixed-Point
Input” on page 9-58.
If you set the class of an input parameter to struct, you must specify the
properties of each field in the structure in the order in which you define the
fields in the structure definition.
Specify fi Class
Sets the input parameter param to the MATLAB class fi (fixed-point numeric
object). For example, to set the class of input U to fi, call:
...
assert(isfi(U));
...
or
...
assert(isa(U,'embedded.fi'));
...
Note If you set the class of an input parameter to fi, you must also set its
numerictype, see “Specify numerictype of Fixed-Point Input” on page 9-57.
You can also set its fimath properties, see “Specify fimath of Fixed-Point
Input” on page 9-58.
9-55
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Sets the input parameter param to the MATLAB class struct (structure). For
example, to set the class of input U to a struct, call:
...
assert(isstruct(U));
...
or
...
assert(isa(U,'struct'));
...
Note If you set the class of an input parameter to struct, you must specify
the properties of each field in the structure in the order in which you define
the fields in the structure definition.
Sets the input parameter param to the size specified by dimensions dims. For
example, to set the size of input U to a 3-by-2 matrix, call:
...
assert(all(size(U)== [3 2]));
...
Sets the size of input parameter param to scalar. For example, to set the
size of input U to scalar, call:
...
assert(isscalar(U));
...
9-56
Define Input Properties Programmatically in MATLAB® File
or
...
assert(all(size(U)== [1]));
...
Specifies that the input parameter param is real. For example, to specify
that input U is real, call:
...
assert(isreal(U));
...
Specifies that the input parameter param is complex. For example, to specify
that input U is complex, call:
...
assert(~isreal(U));
...
...
% Define the numerictype object.
9-57
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
...
% Define the fimath object.
F = fimath('OverflowAction','Saturate');
assert ( function1 ( params ) && function2 ( params ) && function3 ( params ) && ... )
Specifies the class, size, and complexity of one or more inputs using a single
assert function call. For example, the following code specifies that input U is
a double, complex, 3-by-3 matrix, and input V is a 16-bit unsigned integer:
...
assert(isa(U,'double') && ~isreal(U) && all(size(U) == [3 3]) && isa(V,'uint16'));
...
9-58
Define Input Properties Programmatically in MATLAB® File
• Call assert functions at the beginning of the primary function, before any
flow-control operations such as if statements or subroutine calls.
• Do not call assert functions inside conditional constructs, such as if, for,
while, and switch statements.
• If you set the class of an input parameter to fi:
- You must also set its numerictype, see “Specify numerictype of
Fixed-Point Input” on page 9-57.
- You can also set its fimath properties, see “Specify fimath of Fixed-Point
Input” on page 9-58.
• If you set the class of an input parameter to struct, you must specify the
class, size, and complexity of each field in the structure in the order in
which you define the fields in the structure definition.
Property Value
class fi
numerictype numerictype object T, as specified in the
primary function
fimath fimath object F, as specified in the primary
function
size scalar (by default)
complexity real (by default)
function y = emcsqrtfi(x)
T = numerictype('WordLength',32,'FractionLength',23,...
'Signed',true);
F = fimath('SumMode','SpecifyPrecision',...
'SumWordLength',32,'SumFractionLength',23,...
'ProductMode','SpecifyPrecision',...
'ProductWordLength',32,'ProductFractionLength',23);
9-59
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
assert(isfi(x));
assert(isequal(numerictype(x),T));
assert(isequal(fimath(x),F));
y = sqrt(x);
S = struct('r',double(1),'i',fi(4,true,8,0));
This code specifies the class and size of S and its fields when passed as an
input to your MATLAB function:
function y = fcn(S)
y = S;
Note The only way to name a field in a structure is to set at least one of its
properties. Therefore in the preceding example, an assert function specifies
that field S.r is of type double, even though double is the default.
S = struct('r',{double(1), double(2)},'i',...
9-60
Define Input Properties Programmatically in MATLAB® File
{fi(4,1,8,0), fi(5,1,8,0)});
The following code specifies the class and size of each field of structure input S
using the first element of the array:
function y = fcn(S)
y = S;
Note The only way to name a field in a structure is to set at least one of its
properties. Therefore in the example above, an assert function specifies that
field S(1).r is of type double, even though double is the default.
9-61
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Caution For safety, these checks are enabled by default. Without memory
integrity checks, violations will result in unpredictable behavior.
Caution For safety, these checks are enabled by default. Without these
checks the only way to end a long-running execution might be to terminate
MATLAB.
9-62
Control Run-Time Checks
comp_cfg = coder.MEXConfig
comp_cfg.IntegrityChecks = false;
comp_cfg.ExtrinsicCalls = false;
9-63
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
comp_cfg.ResponsivenessChecks = false;
9-64
Fix Run-Time Stack Overflows
9-65
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-66
Code Generation with MATLAB Function Block
For more information about the MATLAB Function block and code generation,
refer to the following:
9-67
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Input Signal or Data Type Override Data Type Override Override Data Type
Parameter Type Setting Applies To Setting
Inherited single Double All numeric types Built-in double
or Floating-point
Single All numeric types Built-in single
or Floating-point
Scaled double All numeric types fi scaled double
or Floating-point
Specified single Double All numeric types Built-in double
or Floating-point
Single All numeric types Built-in single
or Floating-point
Scaled double All numeric types fi scaled double
or Floating-point
Inherited double Double All numeric types Built-in double
or Floating-point
Single All numeric types Built-in single
or Floating-point
Scaled double All numeric types fi scaled double
or Floating-point
Specified double Double All numeric types Built-in double
or Floating-point
Single All numeric types Built-in single
or Floating-point
Scaled double All numeric types fi scaled double
or Floating-point
9-68
Code Generation with MATLAB Function Block
Input Signal or Data Type Override Data Type Override Override Data Type
Parameter Type Setting Applies To Setting
Inherited Fixed Double All numeric types fi double
or Fixed-point
Single All numeric types fi single
or Fixed-point
Scaled double All numeric types fi scaled double
or Fixed-point
Specified Fixed Double All numeric types fi double
or Fixed-point
Single All numeric types fi single
or Fixed-point
Scaled double All numeric types fi scaled double
or Fixed-point
For more information about using the MATLAB Function block with data type
override, see “Using Data Type Override with the MATLAB Function Block”.
Note To simulate models using fixed-point data types in Simulink, you must
have a Fixed-Point Designer license.
9-69
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
1 Place a MATLAB Function block in a new model. You can find the block in
the Simulink User-Defined Functions library.
2 Open the Model Explorer by selecting View > Model Explorer from your
model.
3 Expand the untitled* node in the Model Hierarchy pane of the Model
Explorer. Then, select the MATLAB Function node. The Model Explorer
now appears as shown in the following figure.
9-70
Code Generation with MATLAB Function Block
You can select one of the following options for the MATLAB Function
block fimath:
• Same as MATLAB — When you select this option, the block uses
the same fimath properties as the current default fimath. The edit
box appears dimmed and displays the current default fimath in
read-only form.
• Specify Other — When you select this option, you can specify your
own fimath object in the edit box.
9-71
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
• All fixed-point and integer input signals to the MATLAB Function block
that you choose to treat as fi objects.
• All fi and fimath objects constructed in the MATLAB Function block.
You can set these parameters on the following dialog box, which you can
access through either the Model Explorer or the “Ports and Data Manager”.
9-72
Code Generation with MATLAB Function Block
- Then, select the MATLAB Function block from the Model Hierarchy
pane on the left side of the Model Explorer.
• To access this pane through the Ports and Data Manager, select Tools >
Edit Data/Ports from the MATLAB Editor menu.
When you select Same as MATLAB for the MATLAB Function block
fimath, the MATLAB Function block uses the current default fimath. The
current default fimath appears dimmed and in read-only form in the edit box.
When you select Specify other the block allows you to specify your own
fimath object in the edit box. You can do so in one of two ways:
Note If you use this option and plan to share your model with others,
make sure you define the variable in the model workspace. See “Sharing
Models with Fixed-Point MATLAB Function Blocks” on page 9-73 for more
information on sharing models.
1 Place a MATLAB Function block in a new model. You can find the block in
the Simulink User-Defined Functions library.
2 Define a fimath object in the MATLAB workspace that you want to use for any
Simulink fixed-point signal entering the MATLAB Function block as an input:
F = fimath('RoundingMethod','Floor','OverflowAction','Wrap',...
9-73
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
'ProductMode','KeepLSB','ProductWordLength',32,...
'SumMode','KeepLSB','SumWordLength',32)
F =
RoundingMethod: Floor
OverflowAction: Wrap
ProductMode: KeepLSB
ProductWordLength: 32
SumMode: KeepLSB
SumWordLength: 32
CastBeforeSum: true
3 Open the Model Explorer by selecting View > Model Explorer from your
model.
4 Expand the untitled* node in the Model Hierarchy pane of the Model
Explorer, and select the MATLAB Function node.
5 Select Specify other for the MATLAB Function block fimath parameter
and enter the variable F into the edit box on the Dialog pane. Click Apply
to save your changes.
You have now defined the fimath properties to be associated with all
Simulink fixed-point input signals and all fi and fimath objects constructed
within the block.
6 Select the Base Workspace node in the Model Hierarchy pane. You can see
the variable F that you have defined in the MATLAB workspace listed in the
Contents pane. If you send this model to a coworker, that coworker must first
define that same variable in the MATLAB workspace to get the same results.
7 Cut the variable F from the base workspace, and paste it into the model
workspace listed under the node for your model, in this case, untitled*. The
Model Explorer now appears as shown in the following figure.
9-74
Code Generation with MATLAB Function Block
You can now email your model to a coworker. Because you included the
required variables in the workspace of the model itself, your coworker can
simply run the model and get the correct results. Receiving and running the
model does not require any extra steps.
9-75
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
1 Place a MATLAB Function block in a new model. You can find the block in
the Simulink User-Defined Functions library.
3 Double-click the MATLAB Function block in your model to open the MATLAB
Function Block Editor. Type or copy and paste the following MATLAB code,
including comments, into the Editor:
9-76
Generate Fixed-Point FIR Code Using MATLAB Function Block
b=fidemo.fi_fir_coefficients;
load mtlb
x = mtlb;
n = length(x);
noise = sin(2*pi*2140*(0:n-1)'./Fs);
x = x + noise;
zi = zeros(length(b),1);
9-77
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-78
Generate Fixed-Point FIR Code Using MATLAB Function Block
2 Set the block parameters in the model to these “Fixed-Point FIR Code
Example Parameter Values” on page 9-81.
Parameter Value
Stop time 0
Type Fixed-step
2 Click the cgen_fi > MATLAB Function node in the Model Hierarchy
pane. The dialog box for the MATLAB Function block appears in the Dialog
pane of the Model Explorer.
3 Select Specify other for the MATLAB Function block fimath parameter
on the MATLAB Function block dialog box. You can then create the following
fimath object in the edit box:
fimath('RoundingMethod','Floor','OverflowAction','Wrap',...
'ProductMode','KeepLSB','ProductWordLength',32,...
'SumMode','KeepLSB','SumWordLength',32)
The fimath object you define here is associated with fixed-point inputs to
the MATLAB Function block as well as the fi object you construct within
the block.
By selecting Specify other for the MATLAB Function block fimath, you
ensure that your model always uses the fimath properties you specified.
9-79
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
2 Next, build embeddable C code for your model by selecting the model and
typing Ctrl+B. While the code is building, information outputs to the
MATLAB command line. A folder called coder_fi_grt_rtw is created in
your current working folder.
3 Navigate to coder_fi_grt_rtw > cgen_fi.c. In this file, you can see the code
generated from your model. Search for the following comment in your code:
This search brings you to the beginning of the section of the code that your
MATLAB Function block generated.
9-80
Fixed-Point FIR Code Example Parameter Values
9-81
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-82
Fixed-Point FIR Code Example Parameter Values
9-83
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Variable-size data is data whose size might change at run time. MATLAB
supports bounded and unbounded variable-size data for code generation.
Bounded variable-size data has fixed upper bounds. This data can be allocated
statically on the stack or dynamically on the heap. Unbounded variable-size
data does not have fixed upper bounds. This data must be allocated on
the heap. By default, for MEX and C/C++ code generation, support for
variable-size data is enabled and dynamic memory allocation is enabled for
variable-size arrays whose size exceeds a configurable threshold.
cfg = coder.mexconfig;
cfg.EnableVariableSizing = false;
9-84
Accelerate Code for Variable-Size Data
1 Create a configuration object for code acceleration. For example, for a MEX
function:
mexcfg = coder.mexconfig;
Setting Action
Dynamic memory allocation
mexcfg.DynamicMemoryAllocation='Off'; is disabled. All variable-size
data is allocated statically
on the stack.
Dynamic memory
mexcfg.DynamicMemoryAllocation='AllVariableSizeArrays'; allocation is enabled for
all variable-size arrays.
All variable-size data is
allocated dynamically on the
heap.
Dynamic memory
mexcfg.DynamicMemoryAllocation='Threshold'; allocation is enabled for all
variable-size arrays whose
size (in bytes) is greater than
or equal to the value specified
using the Dynamic memory
allocation threshold
parameter. Variable-size
arrays whose size is less than
this threshold are allocated
on the stack.
9-85
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
This directive:
• Indicates that you intend to generate code for the MATLAB algorithm
• Turns on checking in the MATLAB Code Analyzer to detect potential
errors during code generation
In some cases, the MATLAB Code Analyzer warns you when your code
assigns data a fixed size but later grows the data, such as by assignment
or concatenation in a loop. If that data is supposed to vary in size at run
time, you can ignore these warnings.
For example:
9-86
Accelerate Code for Variable-Size Data
and complexity (real). The second argument, [2 4], indicates the size, a
matrix with two dimensions. The third argument, 1, indicates that the
input is variable sized. The upper bound is 2 for the first dimension and 4
for the second dimension.
9-87
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-88
Accelerate Code for Variable-Size Data
In this function, vector B should expand in size as it adds values from vector A.
Therefore, you can ignore this warning.
The fiaccel function -args option specifies the class, complexity, and size
of each input to function uniquetol:
• The first argument, coder.typeof, defines a variable-size input. The
expression coder.typeof(fi(0,T),[1 100],1) defines input A as a
vector of real, signed embedded.fi objects that have a 16-bit word length
and 15-bit fraction length. The vector has a fixed upper bound; its first
9-89
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-90
Accelerate Code for Variable-Size Data
The error indicates a size mismatch between the left-hand side and right-hand
side of the assignment statement B = [B A(i)];. The assignment B =
A(1) establishes the size of B as a fixed-size scalar (1 x 1). Therefore, the
concatenation of [B A(i)] creates a 1 x 2 vector.
coder.varsize('B');
9-91
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
coder.varsize('B');
B = A(1);
k = 1;
for i = 2:length(A)
if abs(A(k) - A(i)) > tol
B = [B A(i)];
k = i;
end
end
9-92
Accelerate Code for Variable-Size Data
1 Create inputs of the correct class, complexity, and size to pass to the
uniquetol MATLAB and MEX functions.
x = fi(rand(1,90), T);
tol = fi(0, T);
9-93
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
2 Run the original uniquetol function in a loop and time how long it takes
to execute 10 iterations of the loop.
3 Run the generated MEX function with the same inputs for the same
number of loop iterations.
r = tSim/tSim_mex
This example shows that generating a MEX function using fiaccel greatly
accelerates the execution of the fixed-point algorithm.
9-94
Accelerate Fixed-Point Simulation
tempdirObj = fidemo.fiTempdir('fiaccelbasicsdemo');
fiacceldir = tempdirObj.tempDir;
fiaccelsrc = ...
fullfile(matlabroot,'toolbox','fixedpoint','fidemos','+fidemo','fiaccel
copyfile(fiaccelsrc,fiacceldir,'f');
The MATLAB function that performs the feedback loop is in the file
fiaccelFeedback.m. This code quantizes the input, and performs the
feedback loop action :
type(fullfile(fiacceldir,'fiaccelFeedback.m'))
9-95
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
for n = 1:length(x)
y(n) = quantize(x(n) - a*w, true, 16, 12, 'floor', 'wrap');
w = y(n);
end
tic,
y = fiaccelFeedback(x,a,y,w);
t1 = toc;
tic
9-96
Accelerate Fixed-Point Simulation
y2 = fiaccelFeedback_mex(x,y,w);
t2 = toc;
Acceleration Ratio
r = t1/t2
r =
103.2215
clear fiaccelFeedback_mex;
tempdirObj.cleanUp;
9-97
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Prerequisites
• MATLAB
• MATLAB Coder
• Fixed-Point Designer
• C compiler (for most platforms, a default C compiler is
supplied with MATLAB) For a list of supported compilers, see
http://www.mathworks.com/support/compilers/current_release/
You can use mex -setup to change the default compiler. See “Changing
Default Compiler”.
9-98
Propose Fixed-Point Data Types Using an Instrumented Mex Function
y = zeros(size(x));
for i=1:length(x)
y(i) = b(1)*x(i) + z(1);
z(1) = b(2)*x(i) + z(2) - a(2) * y(i);
z(2) = b(3)*x(i) - a(3) * y(i);
end
end
The code generation readiness tool screens the code for features and functions
that are not supported for code generation. The tool reports that the
fun_with_matlab.m function is already suitable for code generation.
9-99
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
If your entry-point function is not suitable for code generation, the tool
provides a report that lists the source files that contain unsupported features
and functions. The report also provides an indication of how much work
you must do to make the MATLAB code ready for code generation. Before
proposing data types, you must fix these issues. For more information, see
“Detect and Debug Code Generation Errors” on page 9-27.
1 Navigate to the work folder that contains the file for this tutorial.
9-100
Propose Fixed-Point Data Types Using an Instrumented Mex Function
3 On the project Overview tab, click the Add files link. Browse to the file
fun_with_matlab.m and then click OK to add the file to the project.
The test script runs the fun_with_matlab function with three input signals:
chirp, step, and impulse. The script then plots the results.
Contents of fun_with_matlab_test
% fun_with_matlab_test
%
% Define representative inputs
N = 256; % Number of points
t = linspace(0,1,N); % Time vector from 0 to 1 second
9-101
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
disp('Test complete.');
The test file runs and displays the outputs of the filter for each of the input
signals.
9-102
Propose Fixed-Point Data Types Using an Instrumented Mex Function
MATLAB Coder determines the input types from the test file and then
displays them in the Autodefine Input Types dialog box.
9-103
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
2 On the Build tab, set the Output type to Instrumented MEX Function.
The Build progress dialog box opens. When the build is complete, MATLAB
Coder generates an instrumented MEX function fun_with_matlab_mex
in the current folder. It also provides a link to the report on the Show
Instrumentation Results pane. In this report, you can view the types of
variables in your MATLAB code.
9-104
Propose Fixed-Point Data Types Using an Instrumented Mex Function
This example uses the default data type proposal settings which propose
fraction lengths for the specified word lengths. Because the MATLAB code
is floating-point, the word length is specified by the Default data type of
all floating-point expressions field. You can specify the numerictype
signedness, word length and fraction length. Specifying [] for signedness
instructs MATLAB Coder to choose the signedness based on simulation
values. The default word length is 16. The default fraction length is 12.
Run Simulation
1 On the Run Simulation pane, verify that the test file is set to
fun_with_matlab_test and that Redirect entry-point calls to MEX
9-105
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
The report displays the simulation minimum and maximum values and the
proposed data types.
MATLAB Coder proposes data types with word length of 16 and fraction
length optimized to avoid overflows.
9-106
Propose Fixed-Point Data Types Using an Instrumented Mex Function
Next Steps
To learn how to apply the proposed data types to your entry-point MATLAB
function and verify that the fixed-point version of your algorithm is
functionally equivalent to your original MATLAB algorithm, see “Apply
Fixed-Point Data Types Using an Instrumented Mex Function” on page 9-108.
9-107
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Prerequisites
• MATLAB
• MATLAB Coder
• Fixed-Point Designer
• C compiler (for most platforms, a default C compiler is
supplied with MATLAB) For a list of supported compilers, see
http://www.mathworks.com/support/compilers/current_release/
You can use mex -setup to change the default compiler. See “Changing
Default Compiler”.
9-108
Apply Fixed-Point Data Types Using an Instrumented Mex Function
9-109
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
y = fi(zeros(size(x)),1,16,14,'OverflowAction','Wrap','RoundingMethod','Floor');
1 Navigate to the work folder that contains the file for this tutorial.
9-110
Apply Fixed-Point Data Types Using an Instrumented Mex Function
3 On the project Overview tab, click the Add files link. Browse to the file
fun_with_fi.m, and then click OK to add the file to the project.
The test file runs and plots the outputs of the filter. MATLAB Coder
determines the input types from the test file and then displays them.
3 In the Autodefine Input Types dialog box, click Use These Types to accept
the autodefined input type.
9-111
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Run Simulation
The fun_with_fi_test file runs. The test file runs the original MATLAB
algorithm and the fixed-point version, and plots the difference in their
outputs.
9-112
Apply Fixed-Point Data Types Using an Instrumented Mex Function
9-113
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-114
Code Generation Readiness Tool
Summary Tab
9-115
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
• MATLAB syntax issues. These issues are reported in the MATLAB editor.
Use the code analyzer to learn more about the issues and how to fix them.
• Unsupported MATLAB function calls.
• Unsupported MATLAB language features, such as recursion, cell arrays,
nested functions, and function handles.
• Unsupported data types.
9-116
Code Generation Readiness Tool
If the code that you are checking calls other MATLAB functions, or you
are checking multiple entry-point functions, the tool displays the Code
Structure Tab.
9-117
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
This tab provides information about the relative size of each file and how
suitable each file is for code generation.
Code Distribution
The Code Distribution pane provides a pie chart that shows the relative
sizes of the files and how suitable each file is for code generation. This
information is useful during the planning phase of a project for estimation
and scheduling purposes. If the report indicates that there are multiple files
not yet suitable for code generation, consider fixing files that require minor
changes before addressing files with significant issues.
Call Tree
The Call Tree pane provides information on the nesting of function calls. For
each called function, the report provides a Code Generation Readiness
score which ranges from 1 to 5. A score of 1 indicates that the tool has
detected issues that require extensive changes to the MATLAB code to make
it suitable for code generation. A score of 5 indicates that the tool has not
detected code generation issues; the code is ready to use with no or minimal
changes. The report also lists the number of lines of code in each file.
9-118
Code Generation Readiness Tool
9-119
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
See Also
• “Check Code Using the Code Generation Readiness Tool” on page 9-121
9-120
Check Code Using the Code Generation Readiness Tool
coder.screener('filename')
The Code Generation Readiness tool opens for the file named filename,
provides a code generation readiness score, and lists issues that must be
fixed prior to code generation.
The Code Generation Readiness tool opens for the selected file and
provides a code generation readiness score and lists issues that must be
fixed prior to code generation.
See Also
• “Code Generation Readiness Tool” on page 9-114
9-121
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
9-122
Fix Errors Detected at Code Generation Time
To fix the errors, modify your MATLAB code to use only those MATLAB
features that are supported for code generation. For more information,
see “Algorithm Design Basics”. Choose a debugging strategy for detecting
and correcting code generation errors in your MATLAB code. For more
information, see “Debugging Strategies” on page 9-27.
If your MATLAB code calls functions on the MATLAB path, unless the code
generation software determines that these functions should be extrinsic or
you declare them to be extrinsic, it attempts to compile these functions. See
“Resolution of Function Calls for Code Generation” on page 11-2. To get
detailed diagnostics, add the %#codegen directive to each external function
that you want codegen to compile.
See Also
• “Use Fixed-Point Code Generation Reports” on page 9-42
• “Why Test MEX Functions in MATLAB?”
• “When to Generate Code from MATLAB Algorithms” on page 16-2
• “Debugging Strategies” on page 9-27
• “Declaring MATLAB Functions as Extrinsic Functions” on page 11-12
9-123
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
Write a simple MATLAB algorithm that adds two numbers and returns the
result.
function y = my_add1(a, b)
y = a+b;
Write a second MATLAB algorithm that adds two numbers using accumpos
and returns the result.
function y =my_add2(a, b)
y = accumpos(a, b); %floor, wrap
accumpos adds a and b using the data type of a. b is cast into the data type of
a. If a is a fi object, by default, accumpos sets the rounding mode to 'Floor'
and the overflow action to 'Wrap'. It ignores the fimath properties of a and b.
a = fi(1.25, 1, 32,5);
b = fi(0.125, 0, 32);
%%
y1 = my_add1(a, b)
y2 = my_add2(a, b)
y1 =
1.3750
9-124
Avoid Multiword Operations in Generated Code
y2 =
1.3750
For the simple addition, the word length grows but using accumpos, the word
length of the result is the same as that of a.
Generate C code for the function my_add1. First, disable use of the long long
data type because it is not usually supported by the target hardware.
hw = coder.HardwareImplementation;
hw.ProdHWDeviceType = 'Generic->32-bit Embedded Processor';
hw.ProdLongLongMode = false;
hw.ProdBitPerLong = 32;
cfg = coder.config('lib');
cfg.HardwareImplementation = hw;
codegen my_add1 -args {a,b} -report -config cfg
MATLAB Coder generates a C static library and provides a link to the code
generation report.
View the generated code for the simple addition. Click the View report link
to open the code generation report and then scroll to the code for the my_add1
function.
/* Function Declarations */
static void MultiWordAdd(const unsigned long u1[], const unsigned long u2[]
unsigned long y[], int n);
static void MultiWordSignedWrap(const unsigned long u1[], int n1, unsigned
n2, unsigned long y[]);
static void sLong2MultiWord(long u, unsigned long y[], int n);
static void sMultiWord2MultiWord(const unsigned long u1[], int n1, unsigned
y[], int n);
static void sMultiWord2sMultiWordSat(const unsigned long u1[], int n1, unsi
long y[], int n);
static void sMultiWordShl(const unsigned long u1[], int n1, unsigned int n2
9-125
9 Code Acceleration and Code Generation from MATLAB® for Fixed-Point Algorithms
View the generated code for the addition using accumpos. Click the View
report link to open the code generation report and then scroll to the code for
the my_add2 function.
9-126
10
10-2
fi Objects with Simulink®
For example, you can use the following code to create a structure in the
MATLAB workspace with a fi object in the values field. You can then use
the From Workspace block to bring the data into a Simulink model.
a = fi([sin(0:10)' sin(10:-1:0)'])
a =
0 -0.5440
0.8415 0.4121
0.9093 0.9893
0.1411 0.6570
-0.7568 -0.2794
-0.9589 -0.9589
-0.2794 -0.7568
0.6570 0.1411
0.9893 0.9093
0.4121 0.8415
-0.5440 0
s.signals.values = a
s =
s.signals.dimensions = 2
s =
s.time = [0:10]'
10-3
10 Interoperability with Other Products
s =
The From Workspace block in the following model has the fi structure s in
the Data parameter.
In the model, the following parameters in the Solver pane of the Model
Configuration Parameters dialog have the indicated settings:
10-4
fi Objects with Simulink®
The To Workspace block writes the result of the simulation to the MATLAB
workspace as a fi structure.
simout.signals.values
ans =
0 -8.7041
13.4634 6.5938
14.5488 15.8296
2.2578 10.5117
-12.1089 -4.4707
-15.3428 -15.3428
10-5
10 Interoperability with Other Products
-4.4707 -12.1089
10.5117 2.2578
15.8296 14.5488
6.5938 13.4634
-8.7041 0
For more information, refer to “Export Signal Data Using Signal Logging” in
the Simulink documentation.
When you log signals from a referenced model or Stateflow® chart in your
model, the word lengths of fi objects may be larger than you expect. The word
lengths of fixed-point signals in referenced models and Stateflow charts are
logged as the next largest data storage container size.
10-6
fi Objects with Simulink®
10-7
10 Interoperability with Other Products
10-8
fi Objects with DSP System Toolbox™
For example, you can use the following code to create a fi object in the
MATLAB workspace. You can then use the Signal From Workspace block to
bring the data into a Simulink model.
a = fi([sin(0:10)' sin(10:-1:0)'])
a =
0 -0.5440
0.8415 0.4121
0.9093 0.9893
0.1411 0.6570
-0.7568 -0.2794
-0.9589 -0.9589
-0.2794 -0.7568
0.6570 0.1411
0.9893 0.9093
0.4121 0.8415
-0.5440 0
The Signal From Workspace block in the following model has these settings:
• Signal — a
• Sample time — 1
• Samples per frame — 2
• Form output after final data value by — Setting to zero
10-9
10 Interoperability with Other Products
The Signal To Workspace block writes the result of the simulation to the
MATLAB workspace as a fi object.
10-10
fi Objects with DSP System Toolbox™
yout =
(:,:,1) =
0.8415 -0.1319
-0.8415 -0.9561
(:,:,2) =
1.0504 1.6463
0.7682 0.3324
(:,:,3) =
-1.7157 -1.2383
0.2021 0.6795
(:,:,4) =
0.3776 -0.6157
-0.9364 -0.8979
(:,:,5) =
1.4015 1.7508
0.5772 0.0678
(:,:,6) =
-0.5440 0
-0.5440 0
10-11
10 Interoperability with Other Products
10-12
Ways to Generate Code
10-13
10 Interoperability with Other Products
10-14
11
11-2
Resolution of Function Calls for Code Generation
Start
Dispatch to Function
Yes on Yes
MATLAB Extrinsic
for execution MATLAB function?
at runtime path?
No No
Yes
Subfunction?
No
No No
Generate
C code
Function
on Yes
MATLAB
path?
No
Generate error
11-3
11 Calling Functions for Code Generation
• Searches two paths, the code generation path and the MATLAB path
See “Compile Path Search Order” on page 11-4.
• Attempts to compile functions unless the code generation software
determines that it should not compile them or you explicitly declare them
to be extrinsic.
If a MATLAB function is not supported for code generation, you can declare
it to be extrinsic by using the construct coder.extrinsic, as described in
“Declaring MATLAB Functions as Extrinsic Functions” on page 11-12.
During simulation, the code generation software generates code for the
call to an extrinsic function, but does not generate the function’s internal
code. Therefore, simulation can run only on platforms where MATLAB
software is installed. During standalone code generation, MATLAB
attempts to determine whether the extrinsic function affects the output of
the function in which it is called — for example by returning mxArrays to
an output variable. Provided that the output does not change, MATLAB
proceeds with code generation, but excludes the extrinsic function from the
generated code. Otherwise, compilation errors occur.
The code generation software detects calls to many common visualization
functions, such as plot, disp, and figure. The software treats these
functions like extrinsic functions but you do not have to declare them
extrinsic using the coder.extrinsic function.
• Resolves file type based on precedence rules described in “Resolution of File
Types on Code Generation Path” on page 11-6
MATLAB searches this path first during code generation. The code
generation path contains the toolbox functions supported for code
generation.
11-4
Resolution of Function Calls for Code Generation
2 MATLAB path
If the function is not on the code generation path, MATLAB searches this
path.
MATLAB applies the same dispatcher rules when searching each path (see
“Function Precedence Order”).
11-5
11 Calling Functions for Code Generation
11-6
Resolution of File Types on Code Generation Path
Start
No
Yes
MEX-file?
No
Yes
Generate MDL-file? Compile
error M-file
No
Yes
P-file?
No
No Yes
M-file?
11-7
11 Calling Functions for Code Generation
....
11-8
Call Local Functions
The following example illustrates how to define and call a local function mean:
len = length(vals);
mean = avg(vals, len);
stdev = sqrt(sum(((vals-avg(vals,len)).^2))/len);
plot(vals,'-+');
11-9
11 Calling Functions for Code Generation
11-10
Call MATLAB® Functions
For example, you might want to call plot to visualize your results in the
MATLAB environment. If you generate a MEX function from a function that
calls plot and then run the generated MEX function, the code generation
software dispatches calls to the plot function to MATLAB. If you generate a
library or executable, the generated code does not contain calls to the plot
function. The code generation report highlights calls from your MATLAB
code to extrinsic functions so that it is easy to determine which functions are
supported only in the MATLAB environment.
11-11
11 Calling Functions for Code Generation
c = sqrt(a^2 + b^2);
create_plot(a, b, color);
coder.extrinsic('patch');
x = [0;a;a];
y = [0;0;b];
patch(x, y, color);
axis('equal');
The code generation software detects that axis is not supported for code
generation and automatically treats it as an extrinsic function. The compiler
does not generate code for patch and axis, but instead dispatches them to
MATLAB for execution.
11-12
Call MATLAB® Functions
2 Click the link to the code generation report and then, in the report, view
the MATLAB code for create_plot.
The report highlights the patch and axis functions to indicate that they
are supported only within the MATLAB environment.
11-13
11 Calling Functions for Code Generation
11-14
Call MATLAB® Functions
In this example, rat and min as treated as extrinsic every time they are
called in the main function foo. There are two ways to narrow the scope of
an extrinsic declaration inside the main function:
function y = mymin(a,b)
coder.extrinsic('min');
y = min(a,b);
11-15
11 Calling Functions for Code Generation
Here, the function rat is extrinsic every time it is called inside the main
function foo, but the function min is extrinsic only when called inside the
local function mymin.
• Call the MATLAB function using feval, as described in “Calling MATLAB
Functions Using feval” on page 11-16.
function y = foo
coder.extrinsic('rat');
[N D] = rat(pi);
y = 0;
y = feval('min', N, D);
11-16
Call MATLAB® Functions
11-17
11 Calling Functions for Code Generation
Here, the top-level function foo calls the extrinsic MATLAB function rat,
which returns two mxArrays representing the numerator N and denominator
D of the rational fraction approximation of pi. Although you can pass these
mxArrays to another MATLAB function — in this case, min — you cannot
assign the mxArray returned by min to the output y.
To fix this problem, define y to be the type and size of the value that you
expect min to return — in this case, a scalar double — as follows:
11-18
Call MATLAB® Functions
y = min(N,D);
• MATLAB functions that inspect the caller, or read or write to the caller’s
workspace do not work during code generation. Such functions include:
- dbstack
- evalin
- assignin
- save
• The MATLAB debugger cannot inspect variables defined in extrinsic
functions.
• Functions in generated code may produce unpredictable results if your
extrinsic function performs the following actions at run time:
- Change folders
- Change the MATLAB path
- Delete or add MATLAB files
- Change warning states
- Change MATLAB preferences
- Change Simulink parameters
11-19
11 Calling Functions for Code Generation
11-20
12
Language Limitations
Although code generation support is provided for common features of classes
such as properties and methods, there are a number of advanced features
which are not supported, such as:
• Events
• Listeners
12-2
MATLAB® Classes Definition for Code Generation
• Arrays of objects
• Recursive data structures
- Linked lists
- Trees
- Graphs
• Overloadable operators subsref, subsassign, and subsindex
In MATLAB, classes can define their own versions of the subsref,
subsassign, and subsindex methods. Code generation does not support
classes that have their own definitions of these methods.
• The empty method
In MATLAB, classes have a built-in static method, empty, which creates an
empty array of the class. Code generation does not support this method.
• The following MATLAB handle class methods:
- addlistener
- delete
- eq
- findobj
- findpro
• The AbortSet property attribute
codegen ClassNameA
12-3
12 Code Generation for MATLAB® Classes
For example, if function foo takes one input, a, that is a MATLAB object,
you cannot generate code for foo by executing:
• Code generation does not support assigning an object of a value class into a
nontunable property. For example, obj.prop=v; is invalid when prop is a
nontunable property and v is an object based on a value class.
• You cannot use coder.extrinsic to declare a class or method as extrinsic.
• You cannot pass a MATLAB class to the function.
• If you use classes in code in the MATLAB Function block, you cannot use
the debugger to view class information.
• The coder.nullcopy function does not support MATLAB classes as inputs.
12-4
MATLAB® Classes Definition for Code Generation
mySystemObject.nonTunableProperty=struct('fieldA','a','fieldB','b');
mySystemObject.nonTunableProperty.fieldA = a;
mySystemObject.nonTunableProperty.fieldB = b;
classdef B < A
methods
function obj = B(varargin)
if nargin == 0
a = 1;
b = 2;
elseif nargin == 1
12-5
12 Code Generation for MATLAB® Classes
a = varargin{1};
b = 1;
elseif nargin == 2
a = varargin{1};
b = varargin{2};
end
obj = obj@A(a,b);
end
end
end
Because the class definition for B uses an if statement before calling the base
class constructor for A, you cannot generate code for function callB:
However, you can generate code for callB if you define class B as:
classdef B < A
methods
function obj = NewB(varargin)
[a,b] = getaandb(varargin{:});
obj = obj@A(a,b);
end
end
end
12-6
MATLAB® Classes Definition for Code Generation
elseif nargin == 2
a = varargin{1};
b = varargin{2};
end
end
12-7
12 Code Generation for MATLAB® Classes
12-8
Generate Code for MATLAB® Value Classes
1 In a writable folder, create a MATLAB value class, Shape. Save the code
as Shape.m.
classdef Shape
% SHAPE Create a shape at coordinates
% centerX and centerY
properties
centerX;
centerY;
end
properties (Dependent = true)
area;
end
methods
function out = get.area(obj)
out = obj.getarea();
end
function obj = Shape(centerX,centerY)
obj.centerX = centerX;
obj.centerY = centerY;
end
end
methods(Abstract = true)
getarea(obj);
end
methods(Static)
function d = distanceBetweenShapes(shape1,shape2)
xDist = abs(shape1.centerX - shape2.centerX);
yDist = abs(shape1.centerY - shape2.centerY);
d = sqrt(xDist^2 + yDist^2);
end
end
end
12-9
12 Code Generation for MATLAB® Classes
2 In the same folder, create a class, Square, that is a subclass of Shape. Save
the code as Square.m.
12-10
Generate Code for MATLAB® Value Classes
7 In the report, on the MATLAB code tab, click the link to the Rhombus class.
The report displays the class definition of the Rhombus class and highlights
the class constructor. On the Variables tab, it provides details of the
variables used in the class. If a variable is a MATLAB object, by default,
the report displays the object without displaying its properties. To view the
complete list of properties, expand the list as shown for obj.
12-11
12 Code Generation for MATLAB® Classes
8 At the top right side of the report, expand the Calls list.
The Calls list shows that there is a call to the Rhombus constructor from
use_shape and that this constructor calls the Shape constructor.
12-12
Generate Code for MATLAB® Value Classes
9 The constructor for the Rhombus class calls the Shape method of the base
Shape class: obj@Shape. In the report, click the Shape link in this call.
12-13
12 Code Generation for MATLAB® Classes
The link takes you to the Shape method in the Shape class definition.
12-14
Generate Code for MATLAB® Handle Classes and System Objects
methods (Access=protected)
% stepImpl method is called by the step method
function y = stepImpl(~,x)
y = x+1;
end
end
end
function y = testAddOne(x)
%#codegen
p = AddOne();
y = p.step(x);
end
12-15
12 Code Generation for MATLAB® Classes
12-16
Generate Code for MATLAB® Handle Classes and System Objects
12-17
12 Code Generation for MATLAB® Classes
For example, consider a base class, Shape that has two specialized subclasses,
Rhombus and Square. The Shape class has an abstract method, getarea,
12-18
MATLAB® Classes in Code Generation Reports
The report displays the properties using just the base property name, not the
fully qualified name. For example, if your code uses variable obj1 that is a
12-19
12 Code Generation for MATLAB® Classes
MATLAB object with property prop1, then the report displays the property as
prop1 not obj1.prop1. When you sort the Variables column, the sort order
is based on the fully qualified property name.
12-20
Troubleshooting Issues with MATLAB® Classes
obj.mymethod().myprop=...
function foo
h = MyClass;
h.mymethod().aa = 12;
12-21
12 Code Generation for MATLAB® Classes
Workaround
Rewrite the code to return the object and then assign a value to a property
of the object.
function foo
h = MyClass;
b=h.mymethod();
b.aa=12;
12-22
13
Note Because
MATLAB does not
support complex
integer arithmetic,
you cannot generate
code for functions that
use complex integer
arithmetic
13-2
Data Definition for Code Generation
13-3
13 Defining Data for Code Generation
Once you set the type and size of a variable, you cannot cast it to another
type or size. In the following example, the variable x is defined as complex
and stays complex:
Mismatches can also occur when you assign a real operand the complex result
of an operation:
13-4
Code Generation for Complex Data
x = 2 + 3i;
y = 2 - 3i;
z = x + y; % z is 4 + 0i.
• Values returned by MEX functions are real when the imaginary part of
the value is zero.
function y = foo()
y = 1 + 0i; % y is complex with imaginary part equal to zero
end
z = foo_mex
function y = foo()
coder.extrinsic('sqrt')
x = 1 + 0i; % x is complex
y = sqrt(x); % x is real, y is real
end
• Functions that take complex arguments but produce real results return
real values.
13-5
13 Defining Data for Code Generation
• Functions that take real arguments but produce complex results return
complex values.
13-6
Code Generation for Characters
13-7
13 Defining Data for Code Generation
13-8
14
When you use varargin and varargout for code generation, there are the
following limitations:
• You cannot use varargout in the function definition for a top-level function.
• You cannot use varargin in the function definition for a top-level function
in a MATLAB Function block in a Simulink model, or in a MATLAB
function in a Stateflow diagram.
• If you use varargin to define an argument to a top-level function, the
code generation software generates the function with a fixed number of
arguments. This fixed number of arguments is based on the number of
example arguments that you provide on the command line or in a MATLAB
Coder project test file.
Common applications of varargin and varargout for code generation are to:
Code generation relies on loop unrolling to produce simple and efficient code
for varargin and varargout. This technique permits most common uses
of varargin and varargout, but some uses are not allowed (see “Variable
Length Argument Lists for Code Generation” on page 14-10).
14-2
Supported Index Expressions
%#codegen
function [x,y,z] = fcn(a,b,c)
[x,y,z] = subfcn(a,b,c);
You can use the following index expressions. The exp arguments must be
constant expressions or depend on a loop index variable.
Expression Description
varargin varargin{exp} Read the value of element
(read only) exp
varargin{exp1: exp2} Read the values of elements
exp1 through exp2
varargin{:} Read the values of all
elements
varargout varargout{exp} Read or write the value of
(read and write) element exp
Note The use of () is not supported for indexing into varargin and
varargout arrays.
14-3
14 Defining Functions for Code Generation
%#codegen
function [cmlen,cmwth,cmhgt] = conv_2_metric(inlen,inwth,inhgt)
[cmlen,cmwth,cmhgt] = inch_2_cm(inlen,inwth,inhgt);
In the following example, the function fcn cannot detect a logical relationship
between the index expression j and the index variable i:
%#codegen
function [x,y,z] = fcn(a,b,c)
[x,y,z] = subfcn(a,b,c);
14-4
Apply Operations to a Variable Number of Arguments
end
As a result, the function does not unroll the loop and generates a compilation
error:
To fix the problem, you can force loop unrolling by wrapping the loop header
in the function coder.unroll, as follows:
%#codegen
function [x,y,z] = fcn(a,b,c)
[x,y,z] = subfcn(a,b,c);
%#codegen
function [cmlen,cmwth,cmhgt] = conv_2_metric(inlen,inwth,inhgt)
[cmlen,cmwth,cmhgt] = inch_2_cm(inlen,inwth,inhgt);
14-5
14 Defining Functions for Code Generation
For more information, see “Variable Length Argument Lists for Code
Generation” on page 14-10.
14-6
Implement Wrapper Functions
%#codegen
function answer = fcn(method,a,b,c)
answer = optimize(method,a,b,c);
• You can use {:} to read all elements of varargin and pass them to another
function.
• You can mix variable and fixed numbers of arguments.
For more information, see “Variable Length Argument Lists for Code
Generation” on page 14-10.
14-7
14 Defining Functions for Code Generation
If Do This:
You assign varargin array elements Verify that for all pairs, the size,
to local variables in the for-loop type, and complexity are the same
for each property and the same for
each value
Properties or values have different Do not assign varargin array
sizes, types, or complexity elements to local variables in a
for-loop; reference the elements
directly
For example, in the following function test1, the sizes of the property strings
and numeric values are not the same in each pair:
%#codegen
function test1
v = create_value('size', 18, 'rgb', [240 9 44]);
end
function v = create_value(varargin)
v = new_value();
for i = 1 : 2 : length(varargin)
name = varargin{i};
value = varargin{i+1};
switch name
case 'size'
v = set_size(v, value);
case 'rgb'
v = set_color(v, value);
otherwise
end
end
end
14-8
Pass Property/Value Pairs
...
Generated code determines the size, type, and complexity of a local variable
based on its first assignment. In this example, the first assignments occur
in the first iteration of the for-loop:
However, in the second iteration, the size of the property string changes to
3 and the size of the numeric value changes to a vector, resulting in a type
mismatch error. To avoid such errors, reference varargin array values
directly, not through local variables, as highlighted in this code:
%#codegen
function test1
v = create_value('size', 18, 'rgb', [240 9 44]);
end
function v = create_value(varargin)
v = new_value();
for i = 1 : 2 : length(varargin)
switch varargin{i}
case 'size'
v = set_size(v, varargin{i+1});
case 'rgb'
v = set_color(v, varargin{i+1});
otherwise
end
end
end
...
14-9
14 Defining Functions for Code Generation
When you use varargin and varargout for code generation, there are the
following limitations:
• You cannot use varargout in the function definition for a top-level function.
• You cannot use varargin in the function definition for a top-level function
in a MATLAB Function block in a Simulink model, or in a MATLAB
function in a Stateflow diagram.
• If you use varargin to define an argument to a top-level function, the
code generation software generates the function with a fixed number of
arguments. This fixed number of arguments is based on the number of
example arguments that you provide on the command line or in a MATLAB
Coder project test file.
%#codegen
function varargout = inch_2_cm(varargin)
for i = 1:length(varargin)
varargout{i} = varargin{i} * 2.54;
end
To fix the problem, write a top-level function that specifies a fixed number
of inputs and outputs. Then call inch_2_cm as an external function or local
function, as in this example:
%#codegen
function [cmL, cmW, cmH] = conv_2_metric(inL, inW, inH)
[cmL, cmW, cmH] = inch_2_cm(inL, inW, inH);
14-10
Variable Length Argument Lists for Code Generation
For code generation, you can use curly braces {}, but not parentheses (),
to index into varargin and varargout arrays. For more information, see
“Supported Index Expressions” on page 14-3.
If you use an expression to index into varargin or varargout, make sure that
the value of the expression can be computed at compile time. For examples,
see “Apply Operations to a Variable Number of Arguments” on page 14-4.
14-11
14 Defining Functions for Code Generation
14-12
15
Defining MATLAB
Variables for C/C++ Code
Generation
For more information, see “Best Practices for Defining Variables for C/C++
Code Generation” on page 15-3.
15-2
Best Practices for Defining Variables for C/C++ Code Generation
Assignment: Defines:
a = 14.7; a as a real double scalar.
Define properties this way so that the variable is defined on the required
execution paths during C/C++ code generation (see Defining a Variable for
Multiple Execution Paths on page 15-4).
The data that you assign to a variable can be a scalar, matrix, or structure. If
your variable is a structure, define the properties of each field explicitly (see
Defining Fields in a Structure on page 15-5).
15-3
15 Defining MATLAB® Variables for C/C++ Code Generation
Initializing the new variable to the value of the assigned data sometimes
results in redundant copies in the generated code. To avoid redundant
copies, you can define variables without initializing their values by using the
coder.nullcopy construct as described in “Eliminate Redundant Copies of
Variables in Generated Code” on page 15-7.
When you define variables, they are local by default; they do not persist
between function calls. To make variables persistent, see “Define and
Initialize Persistent Variables” on page 15-10.
...
if c > 0
x = 11;
end
% Later in your code ...
if c > 0
use(x);
end
...
Here, x is assigned only if c > 0 and used only when c > 0. This code
works in MATLAB, but generates a compilation error during code generation
because it detects that x is undefined on some execution paths (when c <= 0),.
To make this code suitable for code generation, define x before using it:
x = 0;
...
if c > 0
x = 11;
end
% Later in your code ...
if c > 0
use(x);
end
...
15-4
Best Practices for Defining Variables for C/C++ Code Generation
...
if c > 0
s.a = 11;
disp(s);
else
s.a = 12;
s.b = 12;
end
% Try to use s
use(s);
...
Here, the first part of the if statement uses only the field a, and the else
clause uses fields a and b. This code works in MATLAB, but generates a
compilation error during C/C++ code generation because it detects a structure
type mismatch. To prevent this error, do not add fields to a structure after
you perform certain operations on the structure. For more information, see
“Structure Definition for Code Generation” on page 20-2.
To make this code suitable for C/C++ code generation, define all fields of
s before using it.
...
% Define all fields in structure s
s = struct( a ,0, b , 0);
if c > 0
s.a = 11;
disp(s);
else
s.a = 12;
s.b = 12;
end
% Use s
use(s);
...
15-5
15 Defining MATLAB® Variables for C/C++ Code Generation
...
x = 15; % x is of type double by default.
y = uint8(x); % y has the value of x, but cast to uint8.
...
For example, the following initial assignment is not allowed for code
generation:
15-6
Eliminate Redundant Copies of Variables in Generated Code
Note, however, that variable assignments not only copy the properties of the
assigned data to the new variable, but also initialize the new variable to the
assigned value. This forced initialization sometimes results in redundant
copies in C/C++ code. To eliminate redundant copies, define uninitialized
variables by using the coder.nullcopy function, as described in “How to
Eliminate Redundant Copies by Defining Uninitialized Variables” on page
15-7.
When the uninitialized variable is an array, you must initialize all of its
elements before passing the array as an input to a function or operator
— even if the function or operator does not read from the uninitialized
portion of the array.
15-7
15 Defining MATLAB® Variables for C/C++ Code Generation
N = 5;
X = zeros(1,N);
for i = 1:N
if mod(i,2) == 0
X(i) = i;
else
X(i) = 0;
end
end
N = 5;
X = coder.nullcopy(zeros(1,N));
for i = 1:N
if mod(i,2) == 0
X(i) = i;
else
X(i) = 0;
end
end
15-8
Reassignment of Variable Properties
A variable can hold values that have the same class and complexity but
different sizes. If the size of the initial assignment is not constant, the
variable is dynamically sized in generated code. For more information, see
“Variable-Size Data”.
You can reassign the type (class, size, and complexity) of a variable after the
initial assignment if each occurrence of the variable can have only one type.
In this case, the variable is renamed in the generated code to create multiple
independent variables. For more information, see “Reuse the Same Variable
with Different Properties” on page 15-11.
15-9
15 Defining MATLAB® Variables for C/C++ Code Generation
persistent PROD_X;
The definition should appear at the top of the function body, after the
header and comments, but before the first use of the variable. During code
generation, the value of the persistent variable is initialized to an empty
matrix by default. You can assign your own value after the definition by using
the isempty statement, as in this example:
if isempty(PROD_X)
PROD_X = 1;
end
PROD_X = PROD_X * inputvalue;
end
15-10
Reuse the Same Variable with Different Properties
To compile this example and see how MATLAB renames the reused variable t,
see Variable Reuse in an if Statement on page 15-12.
15-11
15 Defining MATLAB® Variables for C/C++ Code Generation
15-12
Reuse the Same Variable with Different Properties
y = sum(u(t(2:end-1)));
end
2 Compile example1.
4 In the MATLAB code pane of the code generation report, place your pointer
over the variable t inside the if statement.
5 In the MATLAB code pane of the report, place your pointer over the
variable t outside the for-loop.
15-13
15 Defining MATLAB® Variables for C/C++ Code Generation
6 Click the Variables tab to view the list of variables used in example1.
The report displays a list of the variables in example1. There are two
uniquely named local variables t>1 and t>2.
• Persistent variables.
• Global variables.
• Variables passed to C code using coder.ref, coder.rref, coder.wref.
• Variables whose size is set using coder.varsize.
• Variables whose names are controlled using coder.cstructname.
• The index variable of a for-loop when it is used inside the loop body.
15-14
Reuse the Same Variable with Different Properties
15-15
15 Defining MATLAB® Variables for C/C++ Code Generation
To avoid this error, use the workarounds provided in the following table.
with:
for k=1:10
• The loop counter decrements by 1 Rewrite the loop so that the end
value is not equal to the minimum
• The end value equals the
value of the integer type. For
minimum value of the integer
example, replace:
type
• The loop is not covering the full N=intmin('int32')
range of the integer type for k=N+10:-1:N
with:
for k=10:-1:1
15-16
Avoid Overflows in for-Loops
M= intmin('int16');
N= intmax('int16');
for k=int32(M):int32(N)
% Loop body
end
• The loop counter increments or Rewrite the loop so that the loop
decrements by a value not equal variable on the last loop iteration is
to 1 equal to the end value.
• On last loop iteration, the loop
variable value is not equal to the
end value
15-17
15 Defining MATLAB® Variables for C/C++ Code Generation
Type Description
char Character array (string)
complex Complex data. Cast function takes real and imaginary
components
double Double-precision floating point
int8, int16, int32, Signed integer
int64
logical Boolean true or false
single Single-precision floating point
struct Structure
uint8, uint16, Unsigned integer
uint32, uint64
Fixed-point See “Fixed-Point Data Types” on page 1-2.
15-18
16
To: Use:
Deploy an application that uses MATLAB Compiler™
handle graphics
Use Java® MATLAB Builder™ JA
Use toolbox functions that do not Toolbox functions that you rewrite for
support code generation desktop and embedded applications
Deploy MATLAB based GUI MATLAB Compiler
applications on a supported
MATLAB host
16-2
When to Generate Code from MATLAB® Algorithms
To: Use:
Deploy web-based or Windows • MATLAB Builder NE
applications
• MATLAB Builder JA
Interface C code with MATLAB MATLAB mex function
16-3
16 Design Considerations for C/C++ Code Generation
16-4
Which Code Generation Feature to Use
16-5
16 Design Considerations for C/C++ Code Generation
16-6
MATLAB® Code Design Considerations for Code Generation
• Data types
C and C++ use static typing. To determine the types of your variables before
use, MATLAB Coder requires a complete assignment to each variable.
• Array sizing
Variable-size arrays and matrices are supported for code generation. You
can define inputs, outputs, and local variables in MATLAB functions to
represent data that varies in size at run time.
• Memory
You can choose whether the generated code uses static or dynamic memory
allocation.
With dynamic memory allocation, you potentially use less memory at the
expense of time to manage the memory. With static memory, you get
better speed, but with higher memory usage. Most MATLAB code takes
advantage of the dynamic sizing features in MATLAB, therefore dynamic
memory allocation typically enables you to generate code from existing
MATLAB code without modifying it much. Dynamic memory allocation also
allows some programs to compile even when upper bounds cannot be found.
Static allocation reduces the memory footprint of the generated code, and
therefore is suitable for applications where there is a limited amount of
available memory, such as embedded applications.
• Speed
Because embedded applications must run in real time, the code must be
fast enough to meet the required clock rate.
To improve the speed of the generated code:
- Choose a suitable C/C++ compiler. Do not use the default compiler that
MathWorks supplies with MATLAB for Windows 32-bit platforms.
- Consider disabling run-time checks.
16-7
16 Design Considerations for C/C++ Code Generation
By default, for safety, the code generated for your MATLAB code
contains memory integrity checks and responsiveness checks. Generally,
these checks result in more generated code and slower simulation.
Disabling run-time checks usually results in streamlined generated code
and faster simulation. Disable these checks only if you have verified that
array bounds and dimension checking is unnecessary.
See Also
• “Data Definition Basics”
• “Variable-Size Data”
• “Bounded Versus Unbounded Variable-Size Data” on page 22-4
• “Control Dynamic Memory Allocation” on page 9-85
• “Control Run-Time Checks” on page 9-62
16-8
Differences in Behavior After Compiling MATLAB® Code
Character Size
MATLAB supports 16-bit characters, but the generated code represents
characters in 8 bits, the standard size for most embedded languages like C.
See “Code Generation for Characters” on page 13-7.
16-9
16 Design Considerations for C/C++ Code Generation
order from the original MATLAB code. Expressions that produce side effects
include those that:
In addition, the generated code does not enforce order of evaluation of logical
operators that do not short circuit.
A = f1() + f2();
as
A = f1();
A = A + f2();
Termination Behavior
Generated code does not match the termination behavior of MATLAB source
code. For example, optimizations remove infinite loops from generated code if
they do not have side effects. As a result, the generated code may terminate
even though the corresponding MATLAB code does not.
16-10
Differences in Behavior After Compiling MATLAB® Code
16-11
16 Design Considerations for C/C++ Code Generation
16-12
MATLAB® Language Features Supported for C/C++ Code Generation
• N-dimensional arrays
• Matrix operations, including deletion of rows and columns
• Variable-sized data (see “Variable-Size Data Definition for Code
Generation” on page 22-3)
• Subscripting (see “Incompatibility with MATLAB in Matrix Indexing
Operations for Code Generation” on page 22-33)
• Complex numbers (see “Code Generation for Complex Data” on page 13-4)
• Numeric classes (see “Supported Variable Types” on page 15-18)
• Double-precision, single-precision, and integer math
• Fixed-point arithmetic (see “Code Acceleration and Code Generation from
MATLAB” on page 9-3)
• Program control statements if, switch, for, while, and break
• Arithmetic, relational, and logical operators
• Local functions
• Persistent variables (see “Define and Initialize Persistent Variables” on
page 15-10)
• Global variables.
• Structures
• Characters (see “Code Generation for Characters” on page 13-7)
• Function handles
• Frames
• Variable length input and output argument lists
• Subset of MATLAB toolbox functions
• MATLAB classes
16-13
16 Design Considerations for C/C++ Code Generation
• Anonymous functions
• Cell arrays
• Java
• Nested functions
• Recursion
• Sparse matrices
• try/catch statements
16-14
17
17-2
Enumerated Types Supported for Code Generation
Syntax
Example
classdef(Enumeration) PrimaryColors < int32
enumeration
Red(1),
Blue(2),
Yellow(4)
end
end
How to Use
Define enumerated data in MATLAB code and compile the source file. For
example, to generate C/C++ code from your MATLAB source, you can use
17-3
17 Code Generation for Enumerated Data
17-4
When to Use Enumerated Data for Code Generation
17-5
17 Code Generation for Enumerated Data
If your function has inputs, you must specify the properties of these inputs
to codegen. For an enumerated data input, use the -args option to pass
one of its allowable values as a sample value. For example, the following
codegen command specifies that the function displayState takes one input
of enumerated data type sysMode.
displayState(sysMode.OFF)
17-6
Generate Code for Enumerated Data from MATLAB® Algorithms
ans =
RED
17-7
17 Code Generation for Enumerated Data
In the MATLAB Command Window, select File > New > Class.
For example, the following code defines an enumerated type called sysMode:
For example, the following code defines a set of two values for enumerated
type sysMode:
17-8
Define Enumerated Data for Code Generation
The name of the file must match the name of the enumerated data type.
The match is case sensitive.
For example, you cannot name an enumerated data type mode because
MATLAB for code generation provides a toolbox function of the same name.
For a list of toolbox functions supported for code generation, see “Functions
and Objects Supported for C and C++ Code Generation — Alphabetical List”
on page 21-2.
17-9
17 Code Generation for Enumerated Data
17-10
Operations on Enumerated Data for Code Generation
Assignment Operator, =
Example Result
RED
17-11
17 Code Generation for Enumerated Data
Cast Operation
Example Result
double(LEDcolor.RED) ans =
z = 2 z =
y = LEDcolor(z)
2
y =
RED
Indexing Operation
Example Result
m = [1 2] m =
n = LEDcolor(m)
p = n(LEDcolor.GREEN) 1 2
n =
GREEN RED
p =
GREEN
17-12
Operations on Enumerated Data for Code Generation
17-13
17 Code Generation for Enumerated Data
This definition must reside on the MATLAB path in a file with the same
name as the class, sysMode.m.
This definition must reside on the MATLAB path in a file called LEDcolor.m.
17-14
Include Enumerated Data in Control Flow Statements
if state == sysMode.ON
led = LEDcolor.GREEN;
else
led = LEDcolor.RED;
end
1 Generate a MEX function for displayState. Use the -args option to pass
one of the allowable values for the enumerated data input as a sample
value.
displayState(sysMode.OFF)
ans =
RED
17-15
17 Code Generation for Enumerated Data
Rewind(4)
end
end
This definition must reside on the MATLAB path in a file with the same
name as the class, VCRState.m.
This definition must reside on the MATLAB path in a file with the same name
as the class, VCRButton.m.
function s = VCR(button)
%#codegen
persistent state
if isempty(state)
state = VCRState.Stop;
end
switch state
case {VCRState.Stop, VCRState.Forward, VCRState.Rewind}
state = handleDefault(button);
case VCRState.Play
switch button
17-16
Include Enumerated Data in Control Flow Statements
1 Generate a MEX function for VCR. Use the -args option to pass one of the
allowable values for the enumerated data input as a sample value.
s = VCR(VCRButton.Stop)
s =
Stop
17-17
17 Code Generation for Enumerated Data
This definition must reside on the MATLAB path in a file with the same
name as the class, State.m.
function s = Setup(initState)
%#codegen
state = initState;
if isempty(state)
state = State.Standby;
end
17-18
Include Enumerated Data in Control Flow Statements
s = state;
function initialize()
% Perform initialization.
function boot()
% Boot the device.
1 Generate a MEX executable for Setup. Use the -args option to pass one of
the allowable values for the enumerated data input as a sample value.
s = Setup(State.Standby)
s =
Ready
17-19
17 Code Generation for Enumerated Data
17-20
Customize Enumerated Types Based on int32
17-21
17 Code Generation for Enumerated Data
Unless you specify otherwise, the default value for an enumerated type is the
first value in the enumerated class definition. To specify a different default
value, add your own getDefaultValue method to the methods section. The
following code shows a shell for the getDefaultValue method:
17-22
Customize Enumerated Types Based on int32
which the method exists. EnumName must be the name of an enumerated value
defined in that class. For example:
This example defines the default as LEDcolor.RED. If this method does not
appear, the default value would be LEDcolor.GREEN, because that is the first
value listed in the enumerated class definition.
function y = getHeaderFile()
% GETHEADERFILE File where type is defined for generated code.
% If specified, this file is #included where required in the code.
% Otherwise, the type is written out in the generated code.
y = 'filename';
end
#include "imported_enum_type.h"
The getHeaderFile method does not create the declaration file itself. You
must provide a file of the specified name that declares the enumerated data
17-23
17 Code Generation for Enumerated Data
type. The file can also contain definitions of enumerated types that you do not
use in your MATLAB code.
methods(Static)
function y=getHeaderFile()
y='my_LEDcolor.h';
end
end
end
3 Generate a library for the function displayState that takes one input
of enumerated data type sysMode.
17-24
Customize Enumerated Types Based on int32
The header file contains a #include statement for the external header file.
#include "my_LEDcolor.h"
17-25
17 Code Generation for Enumerated Data
1 Generate a library for the function displayState that takes one input
of enumerated data type sysMode.
The report displays the header file containing the enumerated data type
definition.
The enumerated value names include the class name prefix LEDcolor_.
17-26
Control Names of Enumerated Type Values in Generated Code
RED(2),
end
methods(Static)
function y=addClassNameToEnumNames()
y=false;
end
end
end
clear classes
7 Open the code generation report and look at the enumerated type definition
in displayState_types.h.
This time the enumerated value names do not include the class name prefix.
• codegen
• “Include Enumerated Data in Control Flow Statements” on page 17-14 for
a description of the example function displayState and its enumerated
type definitions
17-27
17 Code Generation for Enumerated Data
17-28
Restrictions on Use of Enumerated Data in for-Loops
To iterate over a range of enumerated data with consecutive values, you can
cast the enumerated data to int32 in the loop counter.
Because the enumerated values are consecutive, you can use ColorCodes
data in a for-loop like this:
...
for i = int32(ColorCodes.Red):int32(ColorCodes.Purple)
c = ColorCodes(i);
...
end
17-29
17 Code Generation for Enumerated Data
• cast
• cat
• circshift
• flipdim
• fliplr
• flipud
• histc
• ipermute
• isequal
• isequaln
• isfinite
• isinf
• isnan
• issorted
• length
• permute
• repmat
• reshape
• rot90
• shiftdim
• sort
• sortrows
17-30
Toolbox Functions That Support Enumerated Types for Code Generation
• squeeze
17-31
17 Code Generation for Enumerated Data
17-32
18
To generate efficient standalone code for enumerated data, you are restricted
to using a subset of the operations you can perform with function handles in
MATLAB, as described in “Function Handle Limitations for Code Generation”
on page 18-5
18-2
Define and Pass Function Handles for Code Acceleration
disp(m);
function y = map(f,m)
y = m;
for i = 1:numel(y)
y(i) = f(y(i));
end
function y = addone(u)
y = u + 1;
function y = addtwo(u)
y = u + 2;
This code passes function handles @addone and @addtwo to the function map
which increments each element of the matrix m by the amount prescribed
by the referenced function. Note that map stores the function handle in the
input variable f and then uses f to invoke the function — in this case addone
first and then addtwo.
You can use the function fiaccel to convert the function addval to a MEX
executable that you can run in MATLAB. Follow these steps:
18-3
18 Code Generation for Function Handles
m = fi(magic(3));
18-4
Function Handle Limitations for Code Generation
After you bind a variable to a specific function, you cannot use the same
variable to reference two different function handles, as in this example
%Incorrect code
...
x = @plus;
x = @minus;
...
You cannot pass function handles to or from feval and other extrinsic
MATLAB functions. For more information, see “Declaring MATLAB
Functions as Extrinsic Functions” on page 11-12
plot(data, fhandle(data));
x = fhandle(data);
In this example, the function plotFcn receives a function handle and its
data as primary inputs. plotFcn attempts to call the function referenced by
18-5
18 Code Generation for Function Handles
the fhandle with the input data and plot the results. However, this code
generates a compilation error, indicating that the function isa does not
recognize 'function_handle' as a class name when called inside a MATLAB
function to specify properties of primary inputs.
You cannot display or watch function handles from the debugger. They
appear as empty matrices.
18-6
19
19-2
Optimization Strategies
Optimization Strategies
MATLAB Coder introduces certain optimizations when generating C/C++
code or MEX functions from your MATLAB code. For more information, see
“MATLAB Coder Optimizations in Generated Code”.
Condition Action
You have for-loops whose iterations are “Generate Code with Parallel for-Loops
independent of each other. (parfor)”
You have variable-size arrays in your MATLAB “Minimize Dynamic Memory Allocation”
code.
You have multiple variable-size arrays in your “Set Dynamic Memory Allocation Threshold”
MATLAB code. You want dynamical memory
allocation for larger arrays and static allocation
for smaller ones.
You want your generated function to be called “Eliminate Redundant Copies of Function
by reference. Inputs”
You are calling small functions in your “Inline Code”
MATLAB code.
You have limited target memory for your “Control Inlining Using Configuration Object”
generated code. You want to inline small
functions and generate separate code for larger
ones.
You do not want to generate code for “Fold Function Calls into Constants”
expressions that contain constants only.
19-3
19 Generate Efficient and Reusable Code
Condition Action
You have loop operations in your MATLAB “Minimize Redundant Operations in Loops”
code that do not depend on the loop index.
You have integer operations in your MATLAB “Disable Support for Integer Overflow”
code. You know beforehand that integer
overflow will not occur during execution of your
generated code.
You know beforehand that Inf-s and NaN-s will “Disable Support for Non-Finites”
not occur during execution of your generated
code.
You have for-loops with few iterations. “Unroll for-Loops”
You already have legacy C/C++ code optimized “Integrate Custom Code”
for your target environment.
To optimize the memory usage of generated code, for these conditions, perform
the following actions as necessary:
Condition Action
You have if/else/elseif “Prevent Code Generation for
statements or Unused Execution Paths”
switch/case/otherwise statements
in your MATLAB code. You do
not require some branches of the
statements in your generated code.
You have logical array indexing “Rewrite Logical Array Indexing as
in your MATLAB code. For more a Loop”
information, see “Using Logicals in
Array Indexing”.
You want your generated function to “Eliminate Redundant Copies of
be called by reference. Function Inputs”
You have limited stack space for “Control Stack Space Usage”
your generated code.
You are calling small functions in “Inline Code”
your MATLAB code.
19-4
Optimization Strategies
Condition Action
You have limited target memory for “Control Inlining Using
your generated code. You want to Configuration Object”
inline small functions and generate
separate code for larger ones.
You do not want to generate code for “Fold Function Calls into Constants”
expressions that contain constants
only.
You have loop operations in your “Minimize Redundant Operations in
MATLAB code that do not depend on Loops”
the loop index.
You have integer operations in “Disable Support for Integer
your MATLAB code. You know Overflow”
beforehand that integer overflow
will not occur during execution of
your generated code.
You know beforehand that Inf-s and “Disable Support for Non-Finites”
NaN-s will not occur during execution
of your generated code.
19-5
19 Generate Efficient and Reusable Code
Besides streamlining code generation for the original MATLAB code, this
approach also supplies you with C/C++ codes for the individual sections. You
can reuse these codes later by integrating them with other generated C/C++
code using coder.ceval.
19-6
Eliminate Redundant Copies of Function Inputs
...
/* Function Definitions */
void foo(double *A, double B)
{
*A *= B;
}
...
MATLAB generates code that passes the inputs by value and returns the
value of the output:
...
/* Function Definitions */
double foo2(double A, double B)
19-7
19 Generate Efficient and Reusable Code
{
return A * B;
}
...
In some cases, the output of the function cannot be a modified version of its
inputs. If you do not use the inputs later in the function, you can modify
your code to operate on the inputs instead of on a copy of the inputs. One
method is to create additional return values for the function. For example,
consider the code:
function y2=bar(u2)
% Since foo does not use x1 later in the function,
% it would be optimal to do this operation in place
x2=u2.*2;
% The change in dimensions in the following code
% means that it cannot be done in place
y2=[x2,x2];
end
19-8
Inline Code
Inline Code
MATLAB uses internal heuristics to determine whether or not to inline
functions in the generated code. You can use the coder.inline directive to
fine-tune these heuristics for individual functions. For more information,
see coder.inline.
In this section...
“Prevent Function Inlining” on page 19-9
“Use Inlining in Control Flow Statements” on page 19-9
function y = foo(x)
coder.inline('never');
y = x;
end
Suppose you want to generate code for a division function that will be
embedded in a system with limited memory. To optimize memory use in the
generated code, the following function, inline_division, manually controls
inlining based on whether it performs scalar division or vector division:
19-9
19 Generate Efficient and Reusable Code
if any(divisor == 0)
error('Can not divide by 0');
end
y = dividend / divisor;
19-10
Control Inlining Using Configuration Object
• The size of generated code exceeds desired limits due to excessive inlining of
functions. Suppose you include the statement, coder.inline('always'),
inside a certain function. You then call that function at a large number of
different sites in your code. The generated code can be large due to the
function being inlined every time it is called.
The call sites must be different. For instance, inlining does not lead to large
code if the function to be inlined is called several times inside a loop.
• You have limited RAM or stack space.
In this section...
“Control Size of Functions Inlined” on page 19-11
“Control Size of Functions After Inlining” on page 19-12
“Control Stack Size Limit on Inlined Functions” on page 19-12
• In the Project Settings dialog box, on the All Settings tab, set the value of
the field, Inline threshold, to the maximum size that you want.
• At the command line, create a codegen configuration object. Set the value
of the property, InlineThreshold, to the maximum size that you want.
cfg = coder.config('lib');
cfg.InlineThreshold = 100;
19-11
19 Generate Efficient and Reusable Code
• In the Project Settings dialog box, on the All Settings tab, set the value of
the field, Inline threshold max, to the maximum size that you want.
• At the command line, create a codegen configuration object. Set the value
of the property, InlineThresholdMax, to the maximum size that you want.
cfg = coder.config('lib');
cfg.InlineThresholdMax = 100;
You can control the stack size limit on inlined functions from the Project
Settings dialog box or the command line.
• In the Project Settings dialog box, on the All Settings tab, set the value of
the field, Inline stack limit, to the maximum size that you want.
• At the command line, create a codegen configuration object. Set the value
of the property, InlineThresholdMax, to the maximum size that you want.
cfg = coder.config('lib');
cfg.InlineStackLimit = 2000;
19-12
Control Inlining Using Configuration Object
19-13
19 Generate Efficient and Reusable Code
Write a function AddShift that takes an input Shift and adds it to the
elements of a vector. The vector consists of the square of the first 10 natural
numbers. AddShift generates this vector.
Generate code for AddShift using the codegen command. Open the Code
Generation Report.
The code generation software generates code for creating the vector. It adds
Shift to each element of the vector during vector creation. The definition of
AddShift in generated code looks as follows:
y = (1:10).^2+Shift;
with
y = coder.const((1:10).^2)+Shift;
19-14
Fold Function Calls into Constants
Generate code for AddShift using the codegen command. Open the Code
Generation Report.
The code generation software creates the vector containing the squares of the
first 10 natural numbers. In the generated code, it adds Shift to each element
of this vector. The definition of AddShift in generated code looks as follows:
19-15
19 Generate Efficient and Reusable Code
• You have limited stack space, for instance, in case of embedded targets.
• Your C compiler reports a run-time stack overflow.
1 On the Build tab Settings pane, set the Output type to C/C++ Static
Library, C/C++ Dynamic Library, or C/C++ Executable (depending on
your requirements).
2 Click the More settings link to open the Project Settings dialog box.
3 On the Memory tab, set the field, Stack usage max, to the value that
you want.
cfg = coder.config('lib');
cfg.StackUsageMax=400000;
19-16
Stack Allocation and Performance
Stack allocation typically uses memory more efficiently than static allocation.
However, stack space is sometimes limited, typically in embedded processors.
MATLAB Coder allows you to manually set a limit on the stack space usage to
make your generated code suitable for your target hardware. You can choose
this limit based on the target hardware configurations. For more information,
see “Control Stack Space Usage”.
19-17
19 Generate Efficient and Reusable Code
For example, the MATLAB function, foo, uses logical array indexing.
The generated C code for this function is not very efficient. Rewrite the
MATLAB code to use a loop instead of logical indexing:
19-18
Dynamic Memory Allocation and Performance
MATLAB Coder does not provide a size for unbounded arrays in generated
code. Instead, such arrays are referenced indirectly through pointers. For
such arrays, memory cannot be allocated during compilation of generated
code. Based on storage requirements for the arrays, memory is allocated and
freed at run time as required. This run-time allocation and freeing of memory
leads to slower execution of the generated code. For more information on
dynamic memory allocation, see “Bounded Versus Unbounded Variable-Size
Data”.
Instances in the MATLAB code that might lead to dynamic memory allocation
are:
• Array initialization: You specify array size using a variable whose value
is known only at run time.
• After initialization of an array:
- You declare the array as variable-size using coder.varsize without
explicit upper bounds. After this declaration, you expand the array by
concatenation inside a loop. The number of loop runs is known only
at run time.
- You use a reshape function on the array. At least one of the size
arguments to the reshape function is known only at run time.
If you know the maximum possible size of the array, you can avoid dynamic
memory allocation. You can then provide an upper bound for the array and
19-19
19 Generate Efficient and Reusable Code
19-20
Minimize Dynamic Memory Allocation
You can avoid dynamic memory allocation of a variable-size array if you know
its maximum possible size. To do so, follow these steps:
19-21
19 Generate Efficient and Reusable Code
19-22
Provide Maximum Size for Variable-Size Arrays
This code causes dynamic memory allocation because M and N can both
have unbounded negative values. Therefore, their product can be
unbounded and positive even though, individually, their positive values
are bounded.
Tip You can use assert statements to restrict array sizes in most cases.
When expanding an array inside a loop, this strategy does not work if the
number of loop runs is known only at run time.
You can expand arrays beyond their initial size by concatenation. When
you concatenate additional elements inside a loop, there are two syntax
rules for expanding arrays.
1
19-23
19 Generate Efficient and Reusable Code
function avg=RunningAverage(N)
% Initialize average:
19-24
Provide Maximum Size for Variable-Size Arrays
% These will also be the first two elements of the function output
avg=[0 0];
3 By contrast, if you remove the explicit upper bound, the generated code
dynamically allocates avg.
coder.varsize('avg',[1 8]);
19-25
19 Generate Efficient and Reusable Code
with:
coder.varsize('avg');
19-26
Provide Maximum Size for Variable-Size Arrays
mat = [1 2 3 4 5; 4 5 6 7 8]
% Since mat has 10 elements, N must be a factor of 10
% to pass as argument to reshape
out1 = reshape(mat,N,[]);
% N is not restricted
2 Generate code for ReshapeArray using the codegen command (the input
argument does not have to be a factor of 10):
Tip If your system has limited memory, do not use the assert statement
in this way. For an n-element matrix, the assert statement creates an
n-by-n matrix, which might be large.
19-27
19 Generate Efficient and Reusable Code
cfg=coder.config('lib');
cfg.DynamicMemoryAllocation = 'Off';
19-28
Set Dynamic Memory Allocation Threshold
• Disable dynamic memory allocation for smaller arrays. For smaller arrays,
it can be more efficient to speed up generated code by allocating memory
statically. Though static memory allocation can lead to unused storage
space, it might not be a significant consideration for smaller arrays.
• Enable dynamic memory allocation for larger arrays. For larger arrays,
you can reduce storage requirements significantly using dynamic memory
allocation.
3 On the same tab, select the For arrays with max size at or above
threshold option in the Dynamic memory allocation list.
4 Set the Dynamic memory allocation threshold to the value that you
want.
19-29
19 Generate Efficient and Reusable Code
19-30
Set Dynamic Memory Allocation Threshold
cfg = coder.config('lib');
cfg.DynamicMemoryAllocation='Threshold';
cfg.DynamicMemoryAllocationThreshold = 40000;
19-31
19 Generate Efficient and Reusable Code
19-32
Prevent Code Generation for Unused Execution Paths
19-33
19 Generate Efficient and Reusable Code
ch = 's';
The generated C code squares the elements of a 2-by-2 matrix. The choice
variable, ch, and the other branches of the if/elseif/if statement do
not appear in the generated code.
19-34
Prevent Code Generation for Unused Execution Paths
The generated C code finds the sine of the elements of a 2-by-2 matrix. The
variable, flag, and the switch/case/otherwise statement do not appear
in the generated code.
19-35
19 Generate Efficient and Reusable Code
Because you did not specify the maximum number of threads to use, the
generated C code executes the loop iterations in parallel on the available
number of cores.
4 Generate C code for test_parfor. Use -args 0 to specify that the input, u,
is a scalar double. At the MATLAB command line, enter:
In the generated code, the iterations of the parfor-loop run on at most the
number of cores specified by the input, u. If less than u cores are available,
the iterations run on the cores available at the time of the call.
19-36
Generate Code with Parallel for-Loops (parfor)
19-37
19 Generate Efficient and Reusable Code
for i=1:100
C=C + inv(B)*A^i*B;
end
function C=SeriesFunc(A,B,n)
X = coder.typeof(zeros(4));
codegen -config:lib -launchreport SeriesFunc -args {X,X,10}
19-38
Minimize Redundant Operations in Loops
function C=SeriesFunc(A,B,n)
19-39
19 Generate Efficient and Reusable Code
Unroll for-Loops
Unrolling for-loops eliminates the loop logic by creating a separate copy of
the loop body in the generated code for each iteration. Within each iteration,
the loop index variable becomes a constant.
You can also force loop unrolling for individual functions by wrapping the loop
header in a coder.unroll function. For more information, see coder.unroll.
function y = getrand(n)
% Turn off inlining to make
% generated code easier to read
coder.inline('never');
19-40
Unroll for-Loops
The generated C code for getrand(50) does not unroll the for-loop because
the number of iterations is greater than 10:
19-41
19 Generate Efficient and Reusable Code
19-42
Support for Integer Overflow and Non-Finites
• The result of an integer operation falls outside the range that a data type
can represent. This situation is known as integer overflow.
• Non-finite values (inf and NaN) are generated from an operation. The
supporting code is contained in the files rt_nonfinite.c, rtGetInf.c and
rtGetNaN.c (with corresponding header files).
You can suppress generation of the supporting code if you know beforehand
that such situations will not arise. This action reduces the size and increases
the speed of generated code at the cost of potentially producing results that do
not match simulation in case the situations arise.
cfg = coder.config('lib');
19-43
19 Generate Efficient and Reusable Code
cfg.SaturateOnIntegerOverflow = false;
cfg = coder.config('lib');
cfg.SupportNonFinite = false;
19-44
Integrate Custom Code
In such cases, you can integrate your custom code with the code generated by
MATLAB Coder.
This example illustrates how to integrate the function cublasSgemm from the
NVIDIA® CUDA® Basic Linear Algebra Subroutines (CUBLAS) library in
generated code. This function performs matrix multiplication on a Graphics
Processing Unit (GPU).
ExternalLib_API.m
methods (Static)
19-45
19 Generate Efficient and Reusable Code
function tf = isSupportedContext(ctx)
if ctx.isMatlabHostTarget()
tf = true;
else
error('CUBLAS library not available for this target');
end
end
end
19-46
Integrate Custom Code
if(coder.target('MATLAB'))
C=A*B;
else
19-47
19 Generate Efficient and Reusable Code
19-48
Integrate Custom Code
alpha = single(1);
beta = single(0);
end
end
end
end
C= ExternalLib_API.GPU_MatrixMultiply(A,B);
cfg=coder.config('mex');
cfg.TargetLang='C++';
19-49
19 Generate Efficient and Reusable Code
Matrix_Multiply_mex(eye(2,'single'),eye(2,'single'))
19-50
MATLAB® Coder™ Optimizations in Generated Code
In order to improve the execution speed and memory usage of generated code,
MATLAB Coder introduces the following optimizations:
Constant Folding
When possible, the code generation software evaluates expressions in your
MATLAB code that involve compile-time constants only. In the generated
code, it replaces these expressions with the result of the evaluations. This
behavior is known as constant folding. Because of constant folding, the
generated code does not have to evaluate the constants during execution.
Constant folding can occur when the expressions involve scalars only. To
explicitly enforce constant folding of expressions in other cases, use the
19-51
19 Generate Efficient and Reusable Code
coder.const function. For more information, see “Fold Function Calls into
Constants”.
• At the command line, create a configuration object for code generation. Set
the property ConstantFoldingTimeout to the value that you want.
cfg=coder.config('lib');
cfg.ConstantFoldingTimeout = 200;
• In the Project Settings dialog box, on the All Settings tab, set the field
Constant folding timeout to the value that you want.
Loop Fusion
When possible, the code generation software fuses successive loops with
the same number of runs into a single loop in the generated code. This
optimization reduces loop overhead.
The following code contains successive loops, which are fused during code
generation. The function SumAndProduct evaluates the sum and product
of the elements in an array Arr. The function uses two separate loops to
evaluate the sum y_f_sum and product y_f_prod.
The code generated from this MATLAB code evaluates the sum and product
in a single loop.
19-52
MATLAB® Coder™ Optimizations in Generated Code
The following example contains code where successive matrix operations take
place. The function ManipulateMatrix multiplies every element of a matrix
Mat with a factor. To every element in the result, the function then adds a
shift :
function Res=ManipulateMatrix(Mat,factor,shift)
Res=Mat*factor;
Res=Res+shift;
end
The generated code combines the multiplication and addition into a single
loop operation.
19-53
19 Generate Efficient and Reusable Code
The second branch of the if, elseif, else statement is unreachable. If the
variable x is greater than 10, it is also greater than 0. Therefore, the first
branch is executed in preference to the second branch.
MATLAB Coder does not generate code for the unreachable second branch.
19-54
Generate Reusable Code
19-55
19 Generate Efficient and Reusable Code
19-56
20
20-2
Structure Operations Allowed for Code Generation
20-3
20 Code Generation for MATLAB® Structures
...
S = struct('a', 0, 'b', 1, 'c', 2);
p = S;
...
20-4
Define Scalar Structures for Code Generation
if u > 0
x.a = 10;
x.b = 20;
else
x.b = 30; % Generates an error (on variable x)
x.a = 40;
end
y = x.a + x.b;
In this example, the assignment to x.a comes before x.b in the first if
statement clause, but the assignments appear in reverse order in the else
clause. Here is the corrected code:
...
x.c = 10; % Defines structure and creates field c
y = x; % Reads from structure
x.d = 20; % Generates an error
...
20-5
20 Code Generation for MATLAB® Structures
In this example, the attempt to add a new field d after reading from structure
x generates an error.
This restriction extends across the structure hierarchy. For example, you
cannot add a field to a structure after operating on one of its fields or nested
structures, as in this example:
x.c = 10;
y = x.c;
x.d = 20; % Generates an error
In this example, the attempt to add a new field d to structure x after reading
from the structure’s field c generates an error.
20-6
Define Arrays of Structures for Code Generation
Once you have created the array of structures, you can make the structure
fields variable-size using coder.varsize. For more information, see “Declare
a variable-size structure field.”.
2 Call repmat, passing the scalar structure and the dimensions of the array.
For example, the following code creates X, a 1-by-3 array of scalar structures.
Each element of the array is defined by the structure s, which has two fields,
a and b:
...
20-7
20 Code Generation for MATLAB® Structures
s.a = 0;
s.b = 0;
X = repmat(s,1,3);
X(1).a = 1;
X(2).a = 2;
X(3).a = 3;
X(1).b = 4;
X(2).b = 5;
X(3).b = 6;
...
For example, the following code uses concatenation and a local function to
create the elements of a 1-by-3 structure array:
...
W = [ sab(1,2) sab(2,3) sab(4,5) ];
function s = sab(a,b)
s.a = a;
s.b = b;
...
20-8
Make Structures Persistent
if isempty(X)
X.a = 1;
X.b = 2;
end
20-9
20 Code Generation for MATLAB® Structures
For example, the following MATLAB code uses dot notation to index fields
and substructures:
...
substruct1.a1 = 15.2;
substruct1.a2 = int8([1 2;3 4]);
mystruct = struct('ele1',20.5,'ele2',single(100),
'ele3',substruct1);
substruct2 = mystruct;
substruct2.ele3.a2 = 2*(substruct1.a2);
...
To reference the value of a field in a structure array, you must index into the
array to the structure of interest and then reference that structure’s field
individually using dot notation, as in this example:
...
20-10
Index Substructures and Fields
To reference all the values of a particular field for each structure in an array,
use this notation in a for loop, as in this example:
...
s.a = 0;
s.b = 0;
X = repmat(s,1,5);
for i = 1:5
X(i).a = i;
X(i).b = i+1;
end
This example uses the repmat function to define an array of structures, each
with two fields a and b as defined by s. See “Define Arrays of Structures for
Code Generation” on page 20-7 for more information.
20-11
20 Code Generation for MATLAB® Structures
If: Then:
Assigning one structure to another Define each structure with the same
structure. number, type, and size of fields.
Assigning one structure to a Define the structure with the same
substructure of a different structure number, type, and size of fields as
and vice versa. the substructure.
Assigning an element of one The elements must have the same
structure to an element of another type and size.
structure.
The values stored in the fields of a structure are not treated as constant
values in generated code. Therefore, you cannot use field values to set the size
or class of other data. For example, the following code generates a compiler
error if variable-sizing is disabled:
...
Y.a = 3;
Y.b = 5;
X = zeros(Y.a,Y.b); % Generates an error
In this example, even though you set fields a and b of structure Y to the
values 3 and 5 respectively, Y.a and Y.b are not constants in generated code.
Therefore, they are not valid arguments to pass to the function zeros.
20-12
Assign Values to Structures and Fields
...
Y = struct('a',3,'b',5);
X = zeros(Y.a,Y.b); % Generates a fixed-size 3 X 5 matrix
20-13
20 Code Generation for MATLAB® Structures
y = foo(a, S)
[y, S] = foo(a, S)
20-14
21
21-2
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-3
21 Functions, Classes, and System Objects Supported for Code Generation
21-4
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-5
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
bartlett Signal Window length must be a constant.
Processing Expressions or variables are allowed if their
Toolbox values do not change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
beat2range Phased Array Does not support variable-size inputs.
System Toolbox
besselap Signal Filter order must be a constant. Expressions
Processing or variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
beta MATLAB —
betacdf Statistics —
Toolbox™
betainc MATLAB Always returns a complex result.
21-6
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-7
21 Functions, Classes, and System Objects Supported for Code Generation
bitconcat Fixed-Point —
Designer
bitget MATLAB —
bitget Fixed-Point —
Designer
bitmax MATLAB —
bitor MATLAB —
bitor Fixed-Point • Not supported for slope-bias scaled fi
Designer objects.
bitorreduce Fixed-Point —
Designer
bitreplicate Fixed-Point —
Designer
21-8
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-9
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
blackmanharris Signal Window length must be a constant.
Processing Expressions or variables are allowed if their
Toolbox values do not change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
blanks MATLAB —
blkdiag MATLAB —
bohmanwin Signal Window length must be a constant.
Processing Expressions or variables are allowed if their
Toolbox values do not change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
break MATLAB —
21-10
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
butter Signal Filter coefficients must be constants.
Processing Expressions or variables are allowed if their
Toolbox values do not change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
21-11
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
bwlookup Image
Processing • For best results, specify an input image of
Toolbox class logical.
bwmorph Image
Processing • The text string specifying the operation
Toolbox must be a constant and, for best results,
specify an input image of class logical.
bwpack Image Generated code for this function uses a
Processing precompiled platform-specific shared library.
Toolbox
bwselect Image Supports only the 3 and 4 input argument
Processing syntaxes: BW2 = bwselect(BW,c,r) and BW2
Toolbox = bwselect(BW,c,r,n). The optional fourth
input argument, n, must be a compile-time
constant. In addition, with code generation,
bwselect only supports only the 1 and 2 output
argument syntaxes: BW2 = bwselect(___) or
[BW2, idx] = bwselect(___).
Generated code for this function uses a
precompiled platform-specific shared library.
bwunpack Image Generated code for this function uses a
Processing precompiled platform-specific shared library.
Toolbox
21-12
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
char MATLAB —
21-13
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
cheb1ord Signal All inputs must be constants. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
cheb2ap Signal All inputs must be constants. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
21-14
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
chebwin Signal All inputs must be constants. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
cheby1 Signal All inputs must be constants. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
21-15
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
chi2cdf Statistics —
Toolbox
chi2inv Statistics —
Toolbox
chi2pdf Statistics —
Toolbox
chi2rnd Statistics Can return a different sequence of numbers
Toolbox than MATLAB if either of the following is true:
21-16
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-17
21 Functions, Classes, and System Objects Supported for Code Generation
21-18
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-19
21 Functions, Classes, and System Objects Supported for Code Generation
21-20
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-21
21 Functions, Classes, and System Objects Supported for Code Generation
21-22
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-23
21 Functions, Classes, and System Objects Supported for Code Generation
21-24
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-25
21 Functions, Classes, and System Objects Supported for Code Generation
21-26
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-27
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
de2bi Communications —
System Toolbox
deal MATLAB —
deblank MATLAB • Supports only inputs from the char class.
• Input values must be in the range 0-127.
21-28
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-29
21 Functions, Classes, and System Objects Supported for Code Generation
21-30
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-31
21 Functions, Classes, and System Objects Supported for Code Generation
21-32
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
dsp.AdaptiveLattice- DSP System “System Objects in MATLAB Code Generation”
Filter Toolbox
dsp.AffineProjection- DSP System “System Objects in MATLAB Code Generation”
Filter Toolbox
21-33
21 Functions, Classes, and System Objects Supported for Code Generation
21-34
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-35
21 Functions, Classes, and System Objects Supported for Code Generation
21-36
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-37
21 Functions, Classes, and System Objects Supported for Code Generation
21-38
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-39
21 Functions, Classes, and System Objects Supported for Code Generation
21-40
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
ellipap Signal All inputs must be constants. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
21-41
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
end MATLAB —
end Fixed-Point —
Designer
epipolarLine Computer Compile-time constant input: No restrictions.
Vision System Supports MATLAB Function block: Yes
Toolbox
eps MATLAB —
eps Fixed-Point • Supported for scalar fixed-point signals only.
Designer
• Supported for scalar, vector, and matrix, fi
single and fi double signals.
eq MATLAB —
eq Fixed-Point Not supported for fixed-point signals with
Designer different biases.
erf MATLAB —
erfc MATLAB —
erfcinv MATLAB —
erfcx MATLAB —
erfinv MATLAB —
error MATLAB For standalone code generation, excluded from
the generated code.
21-42
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-43
21 Functions, Classes, and System Objects Supported for Code Generation
21-44
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-45
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
21-46
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-47
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
fir2 Signal All inputs must be constants. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
firceqrip DSP System All inputs must be constant. Expressions or
Toolbox variables are allowed if their values do not
change.
21-48
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
fircls1 Signal All inputs must be constants. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
fireqint DSP System All inputs must be constant. Expressions or
Toolbox variables are allowed if their values do not
change.
firgr DSP System • All inputs must be constant. Expressions
Toolbox or variables are allowed if their values do
not change.
• Does not support syntaxes that have cell
array input.
firhalfband DSP System All inputs must be constant. Expressions or
Toolbox variables are allowed if their values do not
change.
21-49
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
firminphase DSP System All inputs must be constant. Expressions or
Toolbox variables are allowed if their values do not
change.
firnyquist DSP System All inputs must be constant. Expressions or
Toolbox variables are allowed if their values do not
change.
firpr2chfb DSP System All inputs must be constant. Expressions or
Toolbox variables are allowed if their values do not
change.
firpm Signal All inputs must be constants. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
21-50
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
fix MATLAB —
fix Fixed-Point —
Designer
fixed.Quantizer Fixed-Point —
Designer
flattopwin Signal All inputs must be constants. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
flintmax MATLAB —
flip MATLAB —
flipdim MATLAB
fliplr MATLAB —
flipud MATLAB —
21-51
21 Functions, Classes, and System Objects Supported for Code Generation
...
fid = fopen(filename, 'r');
if fid == -1
% fopen failed
21-52
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
else
%fopen successful, okay to call fread
A = fread(fid);
...
for MATLAB —
fpdf Statistics —
Toolbox
fprintf MATLAB • Does not support:
- b and t subtypes on %u, %o %x, and %X
formats
- $ flag for reusing input arguments
- printing arrays
• There is no automatic casting. Input
arguments must match their format types
for predictable results.
• Escaped characters are limited to the
decimal range of 0–127.
• A call to fprintf with fileID equal to 1 or
2 becomes printf in the generated C/C++
code in the following cases:
- The fprintf call is inside a parfor loop.
- Extrinsic calls are disabled.
• When the MATLAB behavior differs from
the C compiler behavior, fprintf matches
the C compiler behavior in the following
cases:
- The format specifier has a corresponding
C format specifier, for example, %e or %E.
- The fprintf call is inside a parfor loop.
- Extrinsic calls are disabled.
21-53
21 Functions, Classes, and System Objects Supported for Code Generation
21-54
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
...
N = 100;
[vals,numRead] = fread(fid, N,'*double');
if numRead ~= N
%fewer elements read than expected
end
...
freqspace MATLAB —
freqz Signal When called with no output arguments, and
Processing without a semicolon at the end, freqz returns
Toolbox the complex frequency response of the input
filter, evaluated at 512 points.
If the semicolon is added, the function produces
a plot of the magnitude and phase response of
the filter.
See “freqz With No Output Arguments”.
frnd Statistics Can return a different sequence of numbers
Toolbox than MATLAB if either of the following is true:
21-55
21 Functions, Classes, and System Objects Supported for Code Generation
21-56
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
gcd MATLAB —
ge MATLAB —
ge Fixed-Point • Not supported for fixed-point signals with
Designer different biases.
geocdf Statistics —
Toolbox
geoinv Statistics —
Toolbox
geomean Statistics —
Toolbox
geopdf Statistics —
Toolbox
geornd Statistics Can return a different sequence of numbers
Toolbox than MATLAB if either of the following is true:
21-57
21 Functions, Classes, and System Objects Supported for Code Generation
21-58
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
hankel MATLAB —
21-59
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
harmmean Statistics —
Toolbox
hdlram Fixed-Point —
Designer
hex2dec MATLAB —
hex2num MATLAB • For n = hex2num(S), size(S,2) <=
length(num2hex(0))
hilb MATLAB —
hist MATLAB • Histogram bar plotting not supported; call
with at least one output argument.
• If supplied, the second argument x must be
a scalar constant.
• Inputs must be real.
histc MATLAB • The output of a variable-size array that
becomes a column vector at run time is a
column-vector, not a row-vector.
• If supplied, dim must be a constant.
• “Variable Sizing Restrictions for Code
Generation of Toolbox Functions” on page
22-36
horizonrange Phased Array Does not support variable-size inputs.
System Toolbox
21-60
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
21-61
21 Functions, Classes, and System Objects Supported for Code Generation
21-62
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-63
21 Functions, Classes, and System Objects Supported for Code Generation
21-64
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-65
21 Functions, Classes, and System Objects Supported for Code Generation
21-66
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-67
21 Functions, Classes, and System Objects Supported for Code Generation
21-68
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-69
21 Functions, Classes, and System Objects Supported for Code Generation
21-70
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
intmax MATLAB —
intmin MATLAB —
inv MATLAB Singular matrix inputs can produce nonfinite
values that differ from MATLAB results.
invhilb MATLAB —
ipermute MATLAB “Variable Sizing Restrictions for Code
Generation of Toolbox Functions” on page
22-36
iptcheckconn Image All input arguments must be compile-time
Processing constants.
Toolbox
iqr Statistics —
Toolbox
isa MATLAB —
iscell MATLAB —
ischar MATLAB —
iscolumn MATLAB —
iscolumn Fixed-Point —
Designer
isdeployed MATLAB • Returns true and false as appropriate for
Compiler MEX and SIM targets
• Returns false for other targets
21-71
21 Functions, Classes, and System Objects Supported for Code Generation
21-72
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-73
21 Functions, Classes, and System Objects Supported for Code Generation
21-74
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
kaiserord Signal —
Processing
Toolbox
kron MATLAB —
kurtosis Statistics —
Toolbox
label2rgb Image Referring to the standard syntax:
Processing
Toolbox RGB = label2rgb(L, map, zerocolor, order)
21-75
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
lineToBorderPoints Computer Compile-time constant input: No restrictions.
Vision System Supports MATLAB Function block: Yes
Toolbox
21-76
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-77
21 Functions, Classes, and System Objects Supported for Code Generation
21-78
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-79
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
mdltest Phased Array Does not support variable-size inputs.
System Toolbox
mean MATLAB • Does not support the type parameter.
• If supplied, dim must be a constant.
• “Variable Sizing Restrictions for Code
Generation of Toolbox Functions” on page
22-36
21-80
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-81
21 Functions, Classes, and System Objects Supported for Code Generation
size(coefs,j) = d(j)
size(coefs,m+1) = npieces
21-82
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
size(coefs,m+2) = order
size(coefs,1) = prod(d)*npieces
size(coefs,2) = order
size(coefs,1) = prod(d)*npieces
size(coefs,2) = order
21-83
21 Functions, Classes, and System Objects Supported for Code Generation
21-84
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
mrdivide Fixed-Point —
Designer
MSERRegions Computer Compile-time constant input: No restrictions.
Vision System Supports MATLAB Function block: Yes
Toolbox For code generation, you must specify both the
pixellist cell array and the length of each
array, as the second input. The object outputs,
regions.PixelList as an array. The region
sizes are defined in regions.Lengths.
Generated code for this function uses a
precompiled platform-specific shared library.
mtimes MATLAB “Variable Sizing Restrictions for Code
Generation of Toolbox Functions” on page
22-36
21-85
21 Functions, Classes, and System Objects Supported for Code Generation
21-86
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
nargin MATLAB —
narginchk MATLAB —
nargout MATLAB • For a function with no output arguments,
returns 1 if called without a terminating
semicolon.
nargoutchk MATLAB —
nbincdf Statistics —
Toolbox
nbininv Statistics —
Toolbox
nbinpdf Statistics —
Toolbox
21-87
21 Functions, Classes, and System Objects Supported for Code Generation
21-88
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-89
21 Functions, Classes, and System Objects Supported for Code Generation
21-90
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
21-91
21 Functions, Classes, and System Objects Supported for Code Generation
optimset
21-92
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
pascal MATLAB —
pchip MATLAB • Input x must be strictly increasing.
• Does not remove y entries with NaN values.
• If you generate code for the pp =
pchip(x,y) syntax, you cannot input pp to
the ppval function in MATLAB. To create a
MATLAB pp structure from a pp structure
created by the code generation software:
- In code generation, use unmkpp to return
the piecewise polynomial details to
MATLAB.
- In MATLAB, use mkpp to create the pp
structure.
pdf Statistics —
Toolbox
permute MATLAB “Variable Sizing Restrictions for Code
Generation of Toolbox Functions” on page
22-36
21-93
21 Functions, Classes, and System Objects Supported for Code Generation
21-94
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-95
21 Functions, Classes, and System Objects Supported for Code Generation
21-96
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-97
21 Functions, Classes, and System Objects Supported for Code Generation
21-98
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-99
21 Functions, Classes, and System Objects Supported for Code Generation
21-100
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
polyint MATLAB —
polyval MATLAB —
polyvalm MATLAB —
pow2 Fixed-Point —
Designer
Signal —
pow2db
Processing
Toolbox
21-101
21 Functions, Classes, and System Objects Supported for Code Generation
21-102
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-103
21 Functions, Classes, and System Objects Supported for Code Generation
21-104
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-105
21 Functions, Classes, and System Objects Supported for Code Generation
21-106
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
rdcoupling Phased Array Does not support variable-size inputs.
System Toolbox
rdivide MATLAB —
rdivide Fixed-Point —
Designer
real MATLAB —
real Fixed-Point —
Designer
reallog MATLAB —
realmax MATLAB —
realmax Fixed-Point —
Designer
realmin MATLAB —
realmin Fixed-Point —
Designer
realpow MATLAB —
realsqrt MATLAB —
rectint MATLAB —
21-107
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
reinterpretcast Fixed-Point —
Designer
rem MATLAB • Performs the arithmetic using the output
class. Results might not match MATLAB
due to differences in rounding errors.
• If one of the inputs has type int64 or
uint64, then both inputs must have the
same type.
removefimath Fixed-Point —
Designer
repmat MATLAB —
repmat Fixed-Point —
Designer
resample Signal All inputs must be constant. Expressions or
Processing variables are allowed if their values do not
Toolbox change.
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
rescale Fixed-Point —
Designer
21-108
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-109
21 Functions, Classes, and System Objects Supported for Code Generation
secd MATLAB —
sech MATLAB —
sensorcov Phased Array Does not support variable-size inputs.
System Toolbox
21-110
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-111
21 Functions, Classes, and System Objects Supported for Code Generation
21-112
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
shiftdim MATLAB • Second argument must be a constant.
• “Variable Sizing Restrictions for Code
Generation of Toolbox Functions” on page
22-36
shnidman Phased Array Does not support variable-size inputs.
System Toolbox
sign MATLAB —
sign Fixed-Point —
Designer
sin MATLAB —
sin Fixed-Point —
Designer
sind MATLAB —
single MATLAB —
single Fixed-Point —
Designer
21-113
21 Functions, Classes, and System Objects Supported for Code Generation
21-114
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-115
21 Functions, Classes, and System Objects Supported for Code Generation
21-116
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-117
21 Functions, Classes, and System Objects Supported for Code Generation
21-118
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
tan MATLAB —
tand MATLAB —
tanh MATLAB —
taylorwin Signal Inputs must be constant
Processing
Toolbox Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
tcdf Statistics —
Toolbox
tf2ca DSP System All inputs must be constant. Expressions or
Toolbox variables are allowed if their values do not
change.
tf2cl DSP System All inputs must be constant. Expressions or
Toolbox variables are allowed if their values do not
change.
time2range Phased Array Does not support variable-size inputs.
System Toolbox
21-119
21 Functions, Classes, and System Objects Supported for Code Generation
21-120
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-121
21 Functions, Classes, and System Objects Supported for Code Generation
21-122
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-123
21 Functions, Classes, and System Objects Supported for Code Generation
21-124
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-125
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
uplus MATLAB —
uplus Fixed-Point —
Designer
assert(n<10)
uv2azel Phased Array Does not support variable-size inputs.
System Toolbox
21-126
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-127
21 Functions, Classes, and System Objects Supported for Code Generation
21-128
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-129
21 Functions, Classes, and System Objects Supported for Code Generation
21-130
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-131
21 Functions, Classes, and System Objects Supported for Code Generation
21-132
Functions and Objects Supported for C and C++ Code Generation — Alphabetical List
21-133
21 Functions, Classes, and System Objects Supported for Code Generation
Specifying constants
To specify a constant input for fiaccel, use
coder.Constant. For more information, see
“Specify Constant Inputs at the Command
Line” on page 23-8.
zeros MATLAB • Dimensions must be real, nonnegative,
integers.
zp2tf MATLAB —
zscore Statistics —
Toolbox
21-134
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-135
21 Functions, Classes, and System Objects Supported for Code Generation
In this section...
“MATLAB® Compiler™” on page 21-194
“Matrices and Arrays in MATLAB” on page 21-194
“Nonlinear Numerical Methods in MATLAB” on page 21-202
“Numerical Integration and Differentiation in MATLAB” on page 21-202
“Optimization Functions in MATLAB” on page 21-203
“Phased Array System Toolbox” on page 21-204
“Polynomials in MATLAB” on page 21-212
“Programming Utilities in MATLAB” on page 21-212
“Relational Operators in MATLAB” on page 21-213
“Rounding and Remainder Functions in MATLAB” on page 21-213
“Set Operations in MATLAB” on page 21-214
“Signal Processing in MATLAB” on page 21-218
“Signal Processing Toolbox” on page 21-219
“Special Values in MATLAB” on page 21-224
“Specialized Math in MATLAB” on page 21-224
“Statistics in MATLAB” on page 21-225
“Statistics Toolbox” on page 21-226
“String Functions in MATLAB” on page 21-235
“Structures in MATLAB” on page 21-237
“Trigonometry in MATLAB” on page 21-237
Aerospace Toolbox
C and C++ code generation for the following Aerospace Toolbox quaternion
functions requires the Aerospace Blockset software.
21-136
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-137
21 Functions, Classes, and System Objects Supported for Code Generation
Casting in MATLAB
Function Remarks and Limitations
cast —
char —
class —
21-138
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-139
21 Functions, Classes, and System Objects Supported for Code Generation
lteZadoffChuSeq —
Signal and Delay Management
bi2de —
de2bi —
Display and Visual Analysis
comm.ConstellationDiagram “Code Generation with System Objects and
Functions”
dsp.ArrayPlot “Code Generation with System Objects and
Functions”
dsp.SpectrumAnalyzer “Code Generation with System Objects and
Functions”
dsp.TimeScope “Code Generation with System Objects and
Functions”
Source Coding
comm.DifferentialDecoder “Code Generation with System Objects and
Functions”
comm.DifferentialEncoder “Code Generation with System Objects and
Functions”
Cyclic Redundancy Check Coding
comm.CRCDetector “Code Generation with System Objects and
Functions”
comm.CRCGenerator “Code Generation with System Objects and
Functions”
comm.HDLCRCDetector “Code Generation with System Objects and
Functions”
21-140
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-141
21 Functions, Classes, and System Objects Supported for Code Generation
21-142
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-143
21 Functions, Classes, and System Objects Supported for Code Generation
21-144
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-145
21 Functions, Classes, and System Objects Supported for Code Generation
21-146
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-147
21 Functions, Classes, and System Objects Supported for Code Generation
21-148
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-149
21 Functions, Classes, and System Objects Supported for Code Generation
21-150
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-151
21 Functions, Classes, and System Objects Supported for Code Generation
21-152
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-153
21 Functions, Classes, and System Objects Supported for Code Generation
21-154
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-155
21 Functions, Classes, and System Objects Supported for Code Generation
21-156
Functions and Objects Supported for C and C++ Code Generation — Categorical List
break —
continue —
end —
for —
if, elseif, else —
parfor Treated as a for-loop when used with fiaccel.
return —
switch, case, • If all case expressions are scalar integer values, generates a C switch
otherwise statement. At run time, if the switch value is not an integer, generates
an error.
• When the case expressions contain noninteger or nonscalar values, the
code generation software generates C if statements in place of a C
switch statement.
while —
21-157
21 Functions, Classes, and System Objects Supported for Code Generation
...
fid = fopen(filename, 'r');
if fid == -1
% fopen failed
else
%fopen successful, okay to call fread
A = fread(fid);
...
21-158
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-159
21 Functions, Classes, and System Objects Supported for Code Generation
...
N = 100;
[vals, numRead] = fread(fid, N, '*double');
if numRead ~= N
%fewer elements read than expected
end
...
load
• Use only when generating MEX or code for Simulink simulation. To
load compile-time constants, use coder.load.
• Does not support use of the function without assignment to a structure
or array. For example, use S = load(filename), not load(filename).
21-160
Functions and Objects Supported for C and C++ Code Generation — Categorical List
nargchk
• Output structure does not include stack information.
narginchk —
nargoutchk —
21-161
21 Functions, Classes, and System Objects Supported for Code Generation
21-162
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Estimation
dsp.BurgAREstimator “System Objects in MATLAB Code Generation”
dsp.BurgSpectrumEstimator “System Objects in MATLAB Code Generation”
dsp.CepstralToLPC “System Objects in MATLAB Code Generation”
dsp.CrossSpectrumEstimator “System Objects in MATLAB Code Generation”
21-163
21 Functions, Classes, and System Objects Supported for Code Generation
21-164
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-165
21 Functions, Classes, and System Objects Supported for Code Generation
21-166
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-167
21 Functions, Classes, and System Objects Supported for Code Generation
21-168
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-169
21 Functions, Classes, and System Objects Supported for Code Generation
21-170
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Exponents in MATLAB
Function Remarks and Limitations
exp —
expm —
expm1 —
factorial —
log
• Generates an error during simulation and returns NaN in generated
code when the input value x is real, but the output should be complex.
To get the complex result, make the input value complex by passing
in complex(x).
21-171
21 Functions, Classes, and System Objects Supported for Code Generation
21-172
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Fixed-Point Designer
In addition to function-specific limitations listed in the table, the following
general limitations apply to the use of Fixed-Point Designer functions in
generated code or with fiaccel:
Function Remarks/Limitations
abs N/A
accumneg N/A
21-173
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
accumpos N/A
add • Code generation in MATLAB does not support the syntax
F.add(a,b). You must use the syntax add(F,a,b).
all N/A
any N/A
atan2 N/A
bitand Not supported for slope-bias scaled fi objects.
bitandreduce N/A
bitcmp N/A
bitconcat N/A
bitget N/A
bitor Not supported for slope-bias scaled fi objects.
bitorreduce N/A
bitreplicate N/A
bitrol N/A
bitror N/A
bitset N/A
bitshift N/A
bitsliceget N/A
bitsll Generated code may not handle out of range shifting.
bitsra Generated code may not handle out of range shifting.
bitsrl Generated code may not handle out of range shifting.
bitxor Not supported for slope-bias scaled fi objects.
bitxorreduce N/A
ceil N/A
complex N/A
21-174
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Function Remarks/Limitations
conj N/A
conv • Variable-sized inputs are only supported when the SumMode
property of the governing fimath is set to Specify precision or
Keep LSB.
• For variable-sized signals, you may see different results between
generated code and MATLAB.
- In the generated code, the output for variable-sized signals is
computed using the SumMode property of the governing fimath.
- In MATLAB, the output for variable-sized signals is computed
using the SumMode property of the governing fimath when both
inputs are nonscalar. However, if either input is a scalar,
MATLAB computes the output using the ProductMode of the
governing fimath.
convergent N/A
cordicabs Variable-size signals are not supported.
cordicangle Variable-size signals are not supported.
cordicatan2 Variable-size signals are not supported.
cordiccart2pol Variable-size signals are not supported.
cordiccexp Variable-size signals are not supported.
cordiccos Variable-size signals are not supported.
cordicpol2cart Variable-size signals are not supported.
cordicrotate Variable-size signals are not supported.
cordicsin Variable-size signals are not supported.
cordicsincos Variable-size signals are not supported.
cos N/A
ctranspose N/A
diag If supplied, the index, k, must be a real and scalar integer value that
is not a fi object.
21-175
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
divide • Any non-fi input must be constant; that is, its value must be
known at compile time so that it can be cast to a fi object.
• Complex and imaginary divisors are not supported.
• Code generation in MATLAB does not support the syntax
T.divide(a,b).
double N/A
end N/A
eps • Supported for scalar fixed-point signals only.
• Supported for scalar, vector, and matrix, fi single and fi double
signals.
eq Not supported for fixed-point signals with different biases.
fi • The default constructor syntax without any input arguments is
not supported.
• If the numerictype is not fully specified, the input to fi must be
a constant, a fi, a single, or a built-in integer value. If the input
is a built-in double value, it must be a constant. This limitation
allows fi to autoscale its fraction length based on the known data
type of the input.
• All properties related to data type must be constant for code
generation.
• numerictype object information must be available for
nonfixed-point Simulink inputs.
filter • Variable-sized inputs are only supported when the SumMode
property of the governing fimath is set to Specify precision or
Keep LSB.
21-176
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Function Remarks/Limitations
fimath • Fixed-point signals coming in to a MATLAB Function block from
Simulink are assigned a fimath object. You define this object in
the MATLAB Function block dialog in the Model Explorer.
• Use to create fimath objects in the generated code.
• If the ProductMode property of the fimath object is set to
anything other than FullPrecision, the ProductWordLength and
ProductFractionLength properties must be constant.
• If the SumMode property of the fimath object is set to
anything other than FullPrecision, the SumWordLength and
SumFractionLength properties must be constant.
fix N/A
fixed.Quantizer N/A
floor N/A
ge Not supported for fixed-point signals with different biases.
get The syntax structure = get(o) is not supported.
getlsb N/A
getmsb N/A
gt Not supported for fixed-point signals with different biases.
hdlram N/A
horzcat N/A
imag N/A
int8, int16, int32, N/A
int64
iscolumn N/A
isempty N/A
isequal N/A
isfi N/A
isfimath N/A
21-177
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
isfimathlocal N/A
isfinite N/A
isinf N/A
isnan N/A
isnumeric N/A
isnumerictype N/A
isreal N/A
isrow N/A
isscalar N/A
issigned N/A
isvector N/A
le Not supported for fixed-point signals with different biases.
length N/A
logical N/A
lowerbound N/A
lsb • Supported for scalar fixed-point signals only.
• Supported for scalar, vector, and matrix, fi single and double
signals.
lt Not supported for fixed-point signals with different biases.
max N/A
mean N/A
median N/A
min N/A
minus Any non-fi input must be constant; that is, its value must be known
at compile time so that it can be cast to a fi object.
21-178
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Function Remarks/Limitations
mpower • When the exponent k is a variable and the input is a scalar,
the ProductMode property of the governing fimath must be
SpecifyPrecision.
• When the exponent k is a variable and the input is not
scalar, the SumMode property of the governing fimath must be
SpecifyPrecision.
• Variable-sized inputs are only supported when the SumMode
property of the governing fimath is set to SpecifyPrecision or
Keep LSB.
• For variable-sized signals, you may see different results between
the generated code and MATLAB.
- In the generated code, the output for variable-sized signals is
computed using the SumMode property of the governing fimath.
- In MATLAB, the output for variable-sized signals is computed
using the SumMode property of the governing fimath when
the first input, a, is nonscalar. However, when a is a scalar,
MATLAB computes the output using the ProductMode of the
governing fimath.
mpy • Code generation in MATLAB does not support the syntax
F.mpy(a,b). You must use the syntax mpy(F,a,b).
• When you provide complex inputs to the mpy function inside of a
MATLAB Function block, you must declare the input as complex
before running the simulation. To do so, go to the Ports and
data manager and set the Complexity parameter for all known
complex inputs to On.
mrdivide N/A
21-179
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
mtimes • Any non-fi input must be constant; that is, its value must be
known at compile time so that it can be cast to a fi object.
• Variable-sized inputs are only supported when the SumMode
property of the governing fimath is set to SpecifyPrecision or
KeepLSB.
• For variable-sized signals, you may see different results between
the generated code and MATLAB.
- In the generated code, the output for variable-sized signals is
computed using the SumMode property of the governing fimath.
- In MATLAB, the output for variable-sized signals is computed
using the SumMode property of the governing fimath when both
inputs are nonscalar. However, if either input is a scalar,
MATLAB computes the output using the ProductMode of the
governing fimath.
ndims N/A
ne Not supported for fixed-point signals with different biases.
nearest N/A
numberofelements numberofelements will be removed in a future release. Use numel
instead.
numel N/A
numerictype • Fixed-point signals coming in to a MATLAB Function block from
Simulink are assigned a numerictype object that is populated
with the signal’s data type and scaling information.
• Returns the data type when the input is a nonfixed-point signal.
• Use to create numerictype objects in generated code.
• All numerictype object properties related to the data type must
be constant.
permute N/A
plus Any non-fi inputs must be constant; that is, its value must be known
at compile time so that it can be cast to a fi object.
21-180
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Function Remarks/Limitations
pow2 N/A
power When the exponent k is a variable, the ProductMode property of the
governing fimath must be SpecifyPrecision.
qr N/A
quantize N/A
range N/A
rdivide N/A
real N/A
realmax N/A
realmin N/A
reinterpretcast N/A
removefimath N/A
repmat N/A
rescale N/A
reshape N/A
round N/A
setfimath N/A
sfi • All properties related to data type must be constant for code
generation.
sign N/A
sin N/A
single N/A
size N/A
sort N/A
sqrt • Complex and [Slope Bias] inputs error out.
• Negative inputs yield a 0 result.
storedInteger N/A
21-181
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
storedIntegerToDouble N/A
sub • Code generation in MATLAB does not support the syntax
F.sub(a,b). You must use the syntax sub(F,a,b).
subsasgn N/A
subsref N/A
sum Variable-sized inputs are only supported when the SumMode property
of the governing fimath is set to Specify precision or Keep LSB.
times • Any non-fi input must be constant; that is, its value must be
known at compile time so that it can be cast to a fi object.
• When you provide complex inputs to the times function inside of a
MATLAB Function block, you must declare the input as complex
before running the simulation. To do so, go to the Ports and
data manager and set the Complexity parameter for all known
complex inputs to On.
transpose N/A
tril If supplied, the index, k, must be a real and scalar integer value that
is not a fi object.
triu If supplied, the index, k, must be a real and scalar integer value that
is not a fi object.
ufi • All properties related to data type must be constant for code
generation.
uint8, uint16, uint32, N/A
uint64
uminus N/A
uplus N/A
upperbound N/A
vertcat N/A
21-182
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Histograms in MATLAB
Function Remarks and Limitations
hist • Histogram bar plotting not supported; call with at least one output
argument.
• If supplied, the second argument x must be a scalar constant.
• Inputs must be real.
histc • The output of a variable-size array that becomes a column vector at
run time is a column-vector, not a row-vector.
• If supplied, dim must be a constant.
• “Variable Sizing Restrictions for Code Generation of Toolbox
Functions” on page 22-36
Function Remarks/Limitations
affine2d When generating code, you can only specify singular objects—arrays
of objects are not supported.
bwlookup For best results, specify an input image of class logical.
bwmorph The text string specifying the operation must be a constant and, for
best results, specify an input image of class logical.
bwpack Generated code for this function uses a precompiled platform-specific
shared library.
21-183
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
21-184
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Function Remarks/Limitations
imbothat The input image IM must be either 2-D or 3-D image. The structuring
element input argument SE must be a compile-time constant.
Generated code for this function uses a precompiled platform-specific
shared library.
imclose The input image IM must be either 2-D or 3-D image. The structuring
element input argument SE must be a compile-time constant.
Generated code for this function uses a precompiled platform-specific
shared library.
imcomplement Does not support int64 and uint64 data types.
imdilate The input image IM must be either 2-D or 3-D image. The SE, PACKOPT,
and SHAPE input arguments must be a compile-time constant. The
structuring element argument SE must be a single element—arrays of
structuring elements are not supported. To obtain the same result
as that obtained using an array of structuring elements, call the
function sequentially.
Generated code for this function uses a precompiled platform-specific
shared library.
imerode The input image IM must be either 2-D or 3-D image. The SE, PACKOPT,
and SHAPE input arguments must be a compile-time constant. The
structuring element argument SE must be a single element—arrays of
structuring elements are not supported. To obtain the same result
as that obtained using an array of structuring elements, call the
function sequentially.
Generated code for this function uses a precompiled platform-specific
shared library.
imextendedmax The optional third input argument, conn, must be a compile-time
constant.
Generated code for this function uses a precompiled platform-specific
shared library.
21-185
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
21-186
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Function Remarks/Limitations
21-187
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
• Submit at least two input arguments: the label matrix, L, and the
colormap matrix, map.
• map must be an n-by-3, double, colormap matrix. You cannot use
a string containing the name of a MATLAB colormap function or a
function handle of a colormap function.
• If you set the boundary color zerocolor to the same color as one of
the regions, label2rgb will not issue a warning.
• If you supply a value for order, it must be 'noshuffle'.
mean2 —
padarray Support only up to 3-D inputs.
Input arguments, padval and direction are expected to be
compile-time constants.
projective2d When generating code, you can only specify singular objects—arrays
of objects are not supported.
strel All input arguments must be compile-time constants. The following
methods are not supported for code generation: getsequence,
reflect, translate, disp, display, loadobj. When generating
code, you can only specify singular objects—arrays of objects are not
supported.
21-188
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-189
21 Functions, Classes, and System Objects Supported for Code Generation
21-190
Functions and Objects Supported for C and C++ Code Generation — Categorical List
size(coefs,j) = d(j)
size(coefs,m+1) = npieces
size(coefs,m+2) = order
size(coefs,1) = prod(d)*npieces
size(coefs,2) = order
size(coefs,1) = prod(d)*npieces
size(coefs,2) = order
21-191
21 Functions, Classes, and System Objects Supported for Code Generation
The code generation software does not remove the singleton dimensions.
However, MATLAB might remove singleton dimensions.
For example, suppose that x is a :4-by-:5 array and ppval(pp,0) returns
a 2-by-3 fixed-size array. v has size 2-by-3-by-:4-by-:5. At run time,
suppose that, size(x,1) =1 and size (x,2) = 5. In the generated code, the
size(v) is [2,3,1,5]. In MATLAB, the size is [2,3,5].
rectint —
sph2cart —
spline • Input x must be strictly increasing.
• Does not remove Y entries with NaN values.
• Does not report an error for infinite endslopes in Y.
• If you generate code for the pp = spline(x,Y) syntax, you cannot
input pp to the ppval function in MATLAB. To create a MATLAB pp
structure from a pp structure created by the code generation software:
- In code generation, use unmkpp to return the piecewise polynomial
details to MATLAB.
- In MATLAB, use mkpp to create the pp structure.
unmkpp • pp must be a valid piecewise polynomial structure created by mkpp,
spline, or pchip in MATLAB or by the code generation software.
• Does not support pp structures created by interp1 in MATLAB.
21-192
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-193
21 Functions, Classes, and System Objects Supported for Code Generation
MATLAB Compiler
C and C++ code generation for the following functions requires the MATLAB
Compiler software.
21-194
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-195
21 Functions, Classes, and System Objects Supported for Code Generation
21-196
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Note This limitation does not apply when the input is scalar or a
variable-length row vector.
21-197
21 Functions, Classes, and System Objects Supported for Code Generation
21-198
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-199
21 Functions, Classes, and System Objects Supported for Code Generation
21-200
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-201
21 Functions, Classes, and System Objects Supported for Code Generation
21-202
Functions and Objects Supported for C and C++ Code Generation — Categorical List
optimset
21-203
21 Functions, Classes, and System Objects Supported for Code Generation
21-204
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-205
21 Functions, Classes, and System Objects Supported for Code Generation
21-206
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-207
21 Functions, Classes, and System Objects Supported for Code Generation
21-208
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-209
21 Functions, Classes, and System Objects Supported for Code Generation
21-210
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-211
21 Functions, Classes, and System Objects Supported for Code Generation
Polynomials in MATLAB
Function Remarks and Limitations
poly • Does not discard nonfinite input values
• Complex input produces complex output
• “Variable Sizing Restrictions for Code Generation of Toolbox
Functions” on page 22-36
polyder The output can contain fewer NaNs than the MATLAB output. However,
if the input contains a NaN, the output contains at least one NaN.
polyfit “Variable Sizing Restrictions for Code Generation of Toolbox Functions”
on page 22-36
polyint —
polyval —
polyvalm —
roots • Output is variable size.
• Output is complex.
• Roots are not always in the same order as MATLAB.
• Roots of poorly conditioned polynomials do not always match MATLAB.
21-212
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-213
21 Functions, Classes, and System Objects Supported for Code Generation
21-214
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-215
21 Functions, Classes, and System Objects Supported for Code Generation
21-216
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-217
21 Functions, Classes, and System Objects Supported for Code Generation
- If you specify the 'legacy' option, empty outputs are row vectors,
1-by-0, never 0-by-0.
• When you specify both the 'rows' option and the 'legacy'option,
outputs ia and ic are column vectors. If these outputs are empty, they
are 0-by-1, even if the output C is 0-by-0.
• When the setOrder is 'sorted' or when you specify the 'legacy'
option, the input A must already be sorted in ascending order. The
first output, C, is sorted in ascending order.
• Complex inputs must be single or double.
21-218
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Function Remarks/Limitations
barthannwin Window length must be a constant. Expressions or variables are
allowed if their values do not change.
bartlett Window length must be a constant. Expressions or variables are
allowed if their values do not change.
21-219
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
besselap Filter order must be a constant. Expressions or variables are allowed if
their values do not change.
bitrevorder —
blackman Window length must be a constant. Expressions or variables are
allowed if their values do not change.
blackmanharris Window length must be a constant. Expressions or variables are
allowed if their values do not change.
bohmanwin Window length must be a constant. Expressions or variables are
allowed if their values do not change.
buttap Filter order must be a constant. Expressions or variables are allowed if
their values do not change.
butter Filter coefficients must be constants. Expressions or variables are
allowed if their values do not change.
buttord All inputs must be constants. Expressions or variables are allowed if
their values do not change.
cfirpm All inputs must be constants. Expressions or variables are allowed if
their values do not change.
cheb1ap All inputs must be constants. Expressions or variables are allowed if
their values do not change.
cheb2ap All inputs must be constants. Expressions or variables are allowed if
their values do not change.
cheb1ord All inputs must be constants. Expressions or variables are allowed if
their values do not change.
cheb2ord All inputs must be constants. Expressions or variables are allowed if
their values do not change.
chebwin All inputs must be constants. Expressions or variables are allowed if
their values do not change.
cheby1 All Inputs must be constants. Expressions or variables are allowed if
their values do not change.
cheby2 All inputs must be constants. Expressions or variables are allowed if
their values do not change.
21-220
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Function Remarks/Limitations
db2pow —
dct C and C++ code generation for dct requires DSP System Toolbox
software.
Length of transform dimension must be a power of two. If specified, the
pad or truncation value must be constant. Expressions or variables are
allowed if their values do not change.
downsample —
dpss All inputs must be constants. Expressions or variables are allowed if
their values do not change.
ellip Inputs must be constant. Expressions or variables are allowed if their
values do not change.
ellipap All inputs must be constants. Expressions or variables are allowed if
their values do not change.
ellipord All inputs must be constants. Expressions or variables are allowed if
their values do not change.
filtfilt Filter coefficients must be constants. Expressions or variables are
allowed if their values do not change.
findpeaks —
fir1 All inputs must be constants. Expressions or variables are allowed if
their values do not change.
fir2 All inputs must be constants. Expressions or variables are allowed if
their values do not change.
fircls All inputs must be constants. Expressions or variables are allowed if
their values do not change.
fircls1 All inputs must be constants. Expressions or variables are allowed if
their values do not change.
firls All inputs must be constants. Expressions or variables are allowed if
their values do not change.
firpm All inputs must be constants. Expressions or variables are allowed if
their values do not change.
21-221
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
firpmord All inputs must be constants. Expressions or variables are allowed if
their values do not change.
flattopwin All inputs must be constants. Expressions or variables are allowed if
their values do not change.
freqz When called with no output arguments, and without a semicolon at the
end, freqz returns the complex frequency response of the input filter,
evaluated at 512 points.
If the semicolon is added, the function produces a plot of the magnitude
and phase response of the filter.
See “freqz With No Output Arguments”.
gausswin All inputs must be constant. Expressions or variables are allowed if
their values do not change.
hamming All inputs must be constant. Expressions or variables are allowed if
their values do not change.
hann All inputs must be constant. Expressions or variables are allowed if
their values do not change.
idct C and C++ code generation for idct requires DSP System Toolbox
software.
Length of transform dimension must be a power of two. If specified, the
pad or truncation value must be constant. Expressions or variables are
allowed if their values do not change.
intfilt All inputs must be constant. Expressions or variables are allowed if
their values do not change.
kaiser All inputs must be constant. Expressions or variables are allowed if
their values do not change.
kaiserord —
levinson C and C++ code generation for levinson requires DSP System Toolbox
software.
If specified, the order of recursion must be a constant. Expressions or
variables are allowed if their values do not change.
21-222
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Function Remarks/Limitations
maxflat All inputs must be constant. Expressions or variables are allowed if
their values do not change.
nuttallwin All inputs must be constant. Expressions or variables are allowed if
their values do not change.
parzenwin All inputs must be constant. Expressions or variables are allowed if
their values do not change.
pow2db —
rcosdesign All inputs must be constant. Expressions or variables are allowed if
their values do not change.
rectwin All inputs must be constant. Expressions or variables are allowed if
their values do not change.
resample The upsampling and downsampling factors must be specified as
constants. Expressions or variables are allowed if their values do not
change.
21-223
21 Functions, Classes, and System Objects Supported for Code Generation
Function Remarks/Limitations
upsample Either declare input n as constant, or use the assert function in the
calling function to set upper bounds for n. For example,
assert(n<10)
xcorr —
yulewalk If specified, the order of recursion must be a constant. Expressions or
variables are allowed if their values do not change.
21-224
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Statistics in MATLAB
Function Remarks and Limitations
corrcoef • Row-vector input is only supported when the first two inputs are
vectors and nonscalar.
mean
• Does not support the type parameter.
median
• If supplied, dim must be a constant.
• “Variable Sizing Restrictions for Code Generation of Toolbox
Functions” on page 22-36“Variable Sizing Restrictions for Code
Generation of Toolbox Functions” on page 22-36
mode • Does not support third output argument C (cell array).
• If supplied, dim must be a constant.
• “Variable Sizing Restrictions for Code Generation of Toolbox
Functions” on page 22-36
21-225
21 Functions, Classes, and System Objects Supported for Code Generation
Statistics Toolbox
C and C++ code generation for the following functions requires the Statistics
Toolbox software.
betacdf —
betainv —
betapdf —
betarnd Can return a different sequence of numbers than MATLAB if either of
the following is true:
21-226
Functions and Objects Supported for C and C++ Code Generation — Categorical List
cdf —
chi2cdf —
chi2inv —
chi2pdf —
chi2rnd Can return a different sequence of numbers than MATLAB if either of
the following is true:
21-227
21 Functions, Classes, and System Objects Supported for Code Generation
finv —
fpdf —
frnd Can return a different sequence of numbers than MATLAB if either of
the following is true:
21-228
Functions and Objects Supported for C and C++ Code Generation — Categorical List
gevpdf —
gevrnd Can return a different sequence of numbers than MATLAB if either of
the following is true:
21-229
21 Functions, Classes, and System Objects Supported for Code Generation
iqr —
kurtosis —
logncdf —
logninv —
lognpdf —
lognrnd Can return a different sequence of numbers than MATLAB if either of
the following is true:
21-230
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-231
21 Functions, Classes, and System Objects Supported for Code Generation
ncx2stat —
normcdf —
norminv —
normpdf —
normrnd Can return a different sequence of numbers than MATLAB if either of
the following is true:
21-232
Functions and Objects Supported for C and C++ Code Generation — Categorical List
quantile —
randg —
random —
raylcdf —
raylinv —
raylpdf —
raylrnd Can return a different sequence of numbers than MATLAB if either of
the following is true:
21-233
21 Functions, Classes, and System Objects Supported for Code Generation
unidpdf —
unidrnd Can return a different sequence of numbers than MATLAB if either of
the following is true:
21-234
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-235
21 Functions, Classes, and System Objects Supported for Code Generation
21-236
Functions and Objects Supported for C and C++ Code Generation — Categorical List
Structures in MATLAB
Function Remarks and Limitations
isfield • Does not support cell input for second argument
isstruct —
struct —
Trigonometry in MATLAB
Function Remarks and Limitations
acos • Generates an error during simulation and returns NaN in generated
code when the input value x is real, but the output should be complex.
To get the complex result, make the input value complex by passing
in complex(x).
acosd —
acosh • Generates an error during simulation and returns NaN in generated
code when the input value x is real, but the output should be complex.
To get the complex result, make the input value complex by passing
in complex(x).
acot —
acotd —
acoth —
acsc —
acscd —
acsch —
asec —
asecd —
asech —
21-237
21 Functions, Classes, and System Objects Supported for Code Generation
21-238
Functions and Objects Supported for C and C++ Code Generation — Categorical List
21-239
21 Functions, Classes, and System Objects Supported for Code Generation
21-240
22
function B = nway(A,n)
% Compute average of every N elements of A and put them in B.
if ((mod(numel(A),n) == 0) && (n>=1 && n<=numel(A)))
B = ones(1,numel(A)/n);
k = 1;
for i = 1 : numel(A)/n
B(i) = mean(A(k + (0:n-1)));
k = k + n;
end
else
error('n <= 0 or does not divide number of elements evenly');
end
22-2
Variable-Size Data Definition for Code Generation
By default, the compiler detects code logic that attempts to change these fixed
attributes after initial assignments, and flags these occurrences as errors
during code generation. However, you can override this behavior by defining
variables or structure fields as variable-size data.
22-3
22 Code Generation for Variable-Size Data
You can control the memory allocation of variable-size data. For more
information, see “Control Memory Allocation of Variable-Size Data” on page
22-5.
22-4
Control Memory Allocation of Variable-Size Data
22-5
22 Code Generation for Variable-Size Data
22-6
Specify Variable-Size Data Without Dynamic Memory Allocation
This command specifies that the input to function foo is a matrix of real
doubles with two variable dimensions. The upper bound for the first
dimension is 3; the upper bound for the second dimension is 100. For a
detailed explanation of this syntax, see coder.typeof.
In this example, the input to function foo is a vector of real doubles without
an upper bound.
You do not need to specify upper bounds when using dynamic allocation on
the heap. In this case, MATLAB assumes variable-size data is unbounded
and does not attempt to determine upper bounds.
22-7
22 Code Generation for Variable-Size Data
The second argument of coder.varsize specifies the upper bound for each
instance of the variable specified in the first argument. In this example, the
argument [1 10] indicates that for every instance of Y:
If you are not using dynamic memory allocation, you must also add an assert
statement to provide upper bounds for the dimensions. For example:
22-8
Specify Variable-Size Data Without Dynamic Memory Allocation
22-9
22 Code Generation for Variable-Size Data
22-10
Variable-Size Data in Code Generation Reports
22-11
22 Code Generation for Variable-Size Data
Method See
Assign the data from a variable-size “Using a Matrix Constructor with
matrix constructor such as Nonconstant Dimensions” on page
• ones 22-13
• zeros
• repmat
Assign multiple, constant sizes “Inferring Variable Size from
to the same variable before using Multiple Assignments” on page
(reading) the variable. 22-13
Define all instances of a variable to “Defining Variable-Size Data
be variable sized Explicitly Using coder.varsize” on
page 22-15
22-12
Define Variable-Size Data for Code Generation
If you are not using dynamic memory allocation, you must also add an assert
statement to provide upper bounds for the dimensions. For example:
When dynamic memory allocation is used, MATLAB does not check for upper
bounds; it assumes variable-size data is unbounded.
22-13
22 Code Generation for Variable-Size Data
if (u > 0)
y = ones(3,4,5);
else
y = zeros(3,1);
end
When static allocation is used, this function infers that y is a matrix with
three dimensions, where:
The code generation report represents the size of matrix y like this:
In this case, the code generation report represents the size of matrix y like
this:
22-14
Define Variable-Size Data for Code Generation
coder.varsize('B');
When you supply only the first argument, coder.varsize assumes all
dimensions of B can vary and that the upper bound is size(B).
The third argument specifies which dimensions vary. This argument must be
a logical vector or a double vector containing only zeros and ones. Dimensions
that correspond to zeros or false have fixed size; dimensions that correspond
to ones or true vary in size. coder.varsize usually treats dimensions of size
1 as fixed (see “Defining Variable-Size Matrices with Singleton Dimensions”
on page 22-16).
For more information about the syntax, see the coder.varsize reference
page.
22-15
22 Code Generation for Variable-Size Data
22-16
Define Variable-Size Data for Code Generation
22-17
22 Code Generation for Variable-Size Data
for i = 1:numel(data)
data(i).color = rand-0.5;
data(i).values = 1:i;
end
y = 0;
for i = 1:numel(data)
if data(i).color > 0
y = y + sum(data(i).values);
end;
end
• coder.varsize('data.A(:).B')
In this example, data is a scalar variable that contains matrix A. Each
element of matrix A contains a variable-size field B.
• coder.varsize('data(:).A(:).B')
This expression defines field B inside each element of matrix A inside each
element of matrix data to be variable sized.
22-18
C Code Interface for Arrays
22-19
22 Code Generation for Variable-Size Data
Note that the predefined type corresponding to double is real_T. For more
information on the correspondence between built-in data types and predefined
types in rtwtypes.h, see “How MATLAB Coder Infers C/C++ Data Types”.
22-20
C Code Interface for Arrays
To define two variables, in1 and in2, of this type, use this statement:
Field Description
*data Pointer to data of type <baseType>
*size Pointer to first element of size vector. Length
of the vector equals the number of dimensions.
allocatedSize Number of elements currently allocated for the
array. If the size changes, MATLAB reallocates
memory based on the new size.
numDimensions Number of dimensions of the size vector, that
is, the number of dimensions you can access
without crossing into unallocated or unused
memory
canFreeData Boolean flag indicating how to deallocate
memory:
• true – MATLAB deallocates memory
automatically
• false – Calling program determines when
to deallocate memory
22-21
22 Code Generation for Variable-Size Data
22-22
Diagnose and Fix Variable-Size Data Errors
22-23
22 Code Generation for Variable-Size Data
A(1) = mean(A(:));
if (n == 3)
A = B;
end
Y = A;
22-24
Diagnose and Fix Variable-Size Data Errors
If u < 0
Y = [Y u];
end
When you compile this function, you get an error because y has fixed
dimensions (3 x 3), but x has variable dimensions. Fix this problem by using
explicit indexing to make x the same size as y:
22-25
22 Code Generation for Variable-Size Data
However, compiling this function generates an error because you did not
specify an upper bound for u.
22-26
Incompatibilities with MATLAB® in Variable-Size Support for Code Generation
During code generation, the standard MATLAB scalar expansion rules apply
except when operating on two variable-size expressions. In this case, both
operands must be the same size. The generated code does not perform scalar
expansion even if one of the variable-size expressions turns out to be scalar at
run time. Instead, it generates a size mismatch error at run time for MEX
22-27
22 Code Generation for Variable-Size Data
functions. Run-time error checking does not occur for non-MEX builds; the
generated code will have unspecified behavior.
For example, in the following function, z is scalar for the switch statement
case 0 and case 1. MATLAB applies scalar expansion when evaluating
y(:) = z; for these two cases.
When you generate code for this function, the code generation software
determines that z is variable size with an upper bound of 3.
22-28
Incompatibilities with MATLAB® in Variable-Size Support for Code Generation
If you run the MEX function with u equal to zero or one, even though z is
scalar at run time, the generated code does not perform scalar expansion
and a run-time error occurs.
scalar_exp_test_err1_mex(0)
Sizes mismatch: 9 ~= 1.
Workaround
Use indexing to force z to be a scalar value:
22-29
22 Code Generation for Variable-Size Data
Workarounds
If your application requires generated code to return the same size of
variable-size N-D arrays as MATLAB code, consider one of these workarounds:
B = size(A);
X = B(1:ndims(A));
22-30
Incompatibilities with MATLAB® in Variable-Size Support for Code Generation
end
Workaround
If your application checks whether a matrix is empty, use one of these
workarounds:
• Rewrite your code to use the isempty function instead of the size function.
• Instead of using x=[] to create empty arrays, create empty arrays of a
specific size using zeros. For example:
22-31
22 Code Generation for Variable-Size Data
function y = fun(n)
x = [];
if n > 1
x = ['a', x];
end
y=class(x);
end
fun(0) returns double in MATLAB, but char in the generated code. When
the statement n > 1 is false, MATLAB does not execute x = ['a', x].
The class of x is double, the class of the empty array. However, the code
generation software considers all execution paths. It determines that based
on the statement x = ['a', x], the class of x is char.
Workaround
Instead of using x=[] to create an empty array, create an empty array of
a specific class. For example, use blanks(0) to create an empty array of
characters.
function y = fun(n)
x = blanks(0);
if n > 1
x = ['a', x];
end
y=class(x);
end
In this situation, if the code generation software detects that both A and B are
vectors at compile time, it applies the special rule and gives the same result
22-32
Incompatibilities with MATLAB® in Variable-Size Support for Code Generation
Workaround
Force your data to be a vector by using the colon operator for indexing:
A(B(:)). For example, suppose your code intentionally toggles between
vectors and regular matrices at run time. You can do an explicit check for
vector-vector indexing:
...
if isvector(A) && isvector(B)
C = A(:);
D = C(B(:));
else
D = A(B);
end
...
The indexing in the first branch specifies that C and B(:) are compile-time
vectors. As a result, the code generation software applies the standard
vector-vector indexing rule.
for i = 1:10
M(i) = 5;
end
22-33
22 Code Generation for Variable-Size Data
In this case, the size of M changes as the loop is executed. Code generation
does not support increasing the size of an array over time.
For code generation, preallocate M as highlighted in the following code.
M=zeros(1,10);
for i = 1:10
M(i) = 5;
end
...
M = ones(10,10);
for i=1:10
for j = i:10
M(i,j) = 2 * M(i,j);
end
end
...
Note The matrix M must be defined before entering the loop, as shown in
the highlighted code.
22-34
Incompatibilities with MATLAB® in Variable-Size Support for Code Generation
22-35
22 Code Generation for Variable-Size Data
Common Restrictions
The following common restrictions apply to multiple toolbox functions, but
only for code generation. To determine which of these restrictions apply to
specific library functions, see the table in “Toolbox Functions with Variable
Sizing Restrictions” on page 22-37.
22-36
Variable Sizing Restrictions for Code Generation of Toolbox Functions
Array-to-vector restriction
The function issues an error when a variable-size array that is not a
variable-length vector assumes the shape of a vector at run time. To avoid
the issue, specify the input explicitly as a variable-length vector instead of
a variable-size array.
Array-to-scalar restriction
The function issues an error if a variable-size array assumes a scalar value at
run time. To avoid this issue, specify scalars as fixed size.
22-37
22 Code Generation for Variable-Size Data
22-38
Variable Sizing Restrictions for Code Generation of Toolbox Functions
22-39
22 Code Generation for Variable-Size Data
22-40
Variable Sizing Restrictions for Code Generation of Toolbox Functions
22-41
22 Code Generation for Variable-Size Data
22-42
Variable Sizing Restrictions for Code Generation of Toolbox Functions
22-43
22 Code Generation for Variable-Size Data
22-44
23
Primary Functions
23 Primary Functions
Properties to Specify
If your primary function has inputs, you must specify the following properties
for each input.
23-2
Primary Function Input Specification
• For each field of input structures, specify class, size, and complexity.
• For each field that is fixed-point class, also specify numerictype, and fimath.
Other inputs
Property Default
class double
size scalar
complexity real
numerictype No default
fimath MATLAB default fimath object
Supported Classes
The following table presents the class names supported by Fixed-Point
Designer.
23-3
23 Primary Functions
• You must specify the class of all primary inputs. If you do not specify the
size or complexity of primary inputs, they default to real scalars.
• For each primary function input whose class is fixed point (fi), you must
specify the input numerictype and fimath properties.
• For each primary function input whose class is struct, you must specify
the properties of each of its fields in the order that they appear in the
structure definition.
23-4
Primary Function Input Specification
23-5
23 Primary Functions
• The cell array of sample values must contain the same number of elements
as primary function inputs.
• The order of elements in the cell array must correspond to the order in
which inputs appear in the primary function signature — for example, the
first element in the cell array defines the properties of the first primary
function input.
Note If you specify an empty cell array with the -args option, fiaccel
interprets this to mean that the function takes no inputs; a compile-time error
occurs if the function does have inputs.
23-6
Primary Function Input Specification
• Use a literal cell array of constants to specify that both inputs are real,
scalar, fixed-point values:
• Assign sample values to a cell array variable to specify that both inputs are
real, unsigned 8-bit integer vectors:
a = fi([1;2;3;4],0,8,0)
b = fi([5;6;7;8],0,8,0)
ex = {a,b}
fiaccel -o emcfx emcf -args ex
T = numerictype('WordLength',32,...
'FractionLength',23,'Signed',true);
F = fimath('SumMode','SpecifyPrecision',...
'SumWordLength',32,'SumFractionLength',23,...
'ProductMode','SpecifyPrecision', ...
ProductWordLength',32,'ProductFractionLength',23);
23-7
23 Primary Functions
myeg = { fi(4.0,T,F) };
4 Compile the function sqrtfi using the fiaccel command, passing the
variable myeg as the argument to the-args option, as in this example:
To specify that inputs are constants, use the -args command-line option with
a coder.Constant object. To specify that an input is a constant with the size,
class, complexity, and value of constant_input, use the following syntax:
-args {coder.Constant(constant_input)}
For example, consider the following function identity which copies its input
to its output:
23-8
Primary Function Input Specification
identity(fi(0.1,1,16,15))
ans =
0.1000
identity_mex
The following example shows how to specify that primary input u is a double
scalar variable and primary input p is a constant structure:
u = fi(0.5,1,16,15)
y_m = rowcol(u,tmp)
y_mex = rowcol_mex(u)
23-9
23 Primary Functions
When you enable dynamic memory allocation, you can specify Inf in the size
vector for dimensions with unknown upper bounds at compile time.
coder.extrinsic('error');
Tb = numerictype(1,32,24);
if ((mod(numel(A),n) == 0) && ...
23-10
Primary Function Input Specification
2 Specify the first input A as a fi object. Its first dimension stays fixed in
size and its second dimension can grow to an upper bound of 100. Specify
the second input n as a double scalar.
23-11
23 Primary Functions
In this section...
“How to Use assert with MATLAB® Coder™” on page 23-12
“Rules for Using assert Function” on page 23-19
“Specifying General Properties of Primary Inputs” on page 23-20
“Specifying Properties of Primary Fixed-Point Inputs” on page 23-20
“Specifying Class and Size of Scalar Structure” on page 23-21
“Specifying Class and Size of Structure Array” on page 23-22
You must use one of the following methods when specifying input properties
using the assert function. Use the exact syntax that is provided; do not
modify it.
23-12
Define Input Properties Programmatically in the MATLAB® File
Sets the input parameter param to the MATLAB class class_name. For
example, to set the class of input U to a 32-bit signed integer, call:
...
assert(isa(U,'int32'));
...
If you set the class of an input parameter to fi, you must also set its
numerictype, see “Specify numerictype of Fixed-Point Input” on page 23-17.
You can also set its fimath properties, see “Specify fimath of Fixed-Point
Input” on page 23-18. If you do not set the fimath properties, codegen uses
the MATLAB default fimath value.
If you set the class of an input parameter to struct, you must specify the
properties of all fields in the order that they appear in the structure definition.
Specify fi Class
assert ( isfi ( param ) )
assert ( isa ( param, 'embedded.fi' ) )
Sets the input parameter param to the MATLAB class fi (fixed-point numeric
object). For example, to set the class of input U to fi, call:
...
assert(isfi(U));
...
or
23-13
23 Primary Functions
...
assert(isa(U,'embedded.fi'));
...
If you set the class of an input parameter to fi, you must also set its
numerictype, see “Specify numerictype of Fixed-Point Input” on page 23-17.
You can also set its fimath properties, see “Specify fimath of Fixed-Point
Input” on page 23-18. If you do not set the fimath properties, codegen uses
the MATLAB default fimath value.
Sets the input parameter param to the MATLAB class struct (structure). For
example, to set the class of input U to a struct, call:
...
assert(isstruct(U));
...
or
...
assert(isa(U, 'struct'));
...
If you set the class of an input parameter to struct, you must specify the
properties of all fields in the order they appear in the structure definition.
Sets the input parameter param to the size specified by dimensions dims. For
example, to set the size of input U to a 3-by-2 matrix, call:
...
assert(all(size(U)== [3 2]));
...
23-14
Define Input Properties Programmatically in the MATLAB® File
Sets the size of input parameter param to scalar. To set the size of input
U to scalar, call:
...
assert(isscalar(U));
...
or
...
assert(all(size(U)== [1]));
...
Sets the upper-bound size of each dimension of input parameter param. To set
the upper-bound size of input U to be less than or equal to a 3-by-2 matrix, call:
assert(all(size(U)<=[3 2]));
Note You can also specify upper bounds for variable-size inputs using
coder.varsize.
23-15
23 Primary Functions
• You must also specify an upper-bound size for each dimension of the input
parameter.
• For each dimension, k, the lower-bound Mk must be less than or equal to
the upper-bound Nk.
• To specify a fixed-size dimension, set the lower and upper bound of a
dimension to the same value.
• Bounds must be non-negative.
To fix the size of the first dimension of input U to 3 and set the second
dimension as variable size with upper-bound of 2, call:
assert(all(size(U)>=[3 0]));
assert(all(size(U)<=[3 2]));
assert(size(U,1)<=3)
assert(size(U,2)==2)
23-16
Define Input Properties Programmatically in the MATLAB® File
Specifies that the input parameter param is real. To specify that input U is
real, call:
...
assert(isreal(U));
...
Specifies that the input parameter param is complex. To specify that input U
is complex, call:
...
assert(~isreal(U));
...
%#codegen
...
% Define the numerictype object.
T = numerictype(1, 32, 30);
23-17
23 Primary Functions
%#codegen
...
% Define the fimath object.
F = fimath('OverflowMode','saturate');
If you do not specify the fimath properties using assert, codegen uses the
MATLAB default fimath value.
Specifies the class, size, and complexity of one or more inputs using a single
assert function call. For example, the following code specifies that input U is
a double, complex, 3-by-3 matrix, and input V is a 16-bit unsigned integer:
%#codegen
...
assert(isa(U,'double') &&
~isreal(U) &&
all(size(U) == [3 3]) &&
isa(V,'uint16'));
...
23-18
Define Input Properties Programmatically in the MATLAB® File
23-19
23 Primary Functions
%#codegen
function y = mcspecgram(pennywhistle,win)
nx = 220500;
nfft = 1024;
assert(isa(pennywhistle,'int16'));
assert(all(size(pennywhistle) == [nx 1]));
assert(isa(win, 'double'));
assert(all(size(win) == [nfft 1]));
...
Alternatively, you can combine property specifications for one or more inputs
inside assert commands:
%#codegen
function y = mcspecgram(pennywhistle,win)
nx = 220500;
nfft = 1024;
assert(isa(pennywhistle,'int16') && all(size(pennywhistle) == [nx 1]));
assert(isa(win, 'double') && all(size(win) == [nfft 1]));
...
23-20
Define Input Properties Programmatically in the MATLAB® File
In the following example, the primary MATLAB function mcsqrtfi takes one
fixed-point input x. The code specifies the following properties for this input.
Property Value
class fi
numerictype numerictype object T, as specified in the
primary function
fimath fimath object F, as specified in the primary
function
size scalar
complexity real (by default)
y = sqrt(x);
S = struct('r',double(1),'i',int8(4));
Here is code that specifies the class and size of S and its fields when passed as
an input to your MATLAB function:
23-21
23 Primary Functions
In most cases, when you don’t explicitly specify values for properties,
MATLAB Coder uses defaults — except for structure fields. The only way
to name a field in a structure is to set at least one of its properties. As a
minimum, you must specify the class of a structure field
The following code specifies the class and size of each field of structure input S
using the first element of the array:
%#codegen
function y = fcn(S)
The only way to name a field in a structure is to set at least one of its
properties. As a minimum, you must specify the class of all fields.
23-22
24
24-2
Code Generation for System Objects
24-3
24 System Objects Supported for Code Generation
24-4
25
System Objects
25-2
Create System Objects
flag = isChangedProperty(H,'Normalize')
The output data from the step method is stored in Y, which, in this case, is
port input and output data.
25-3
25 System Objects
To display a particular property value, use the handle of the created object
followed by the property name: <handle>.<Name>.
Example
This example retrieves and displays the RAMType property value for the
previously created hdlram object:
H.RAMType
25-4
Set Up System Objects
Note When you use Name-Value pair syntax, the object sets property values
in the order you list them. If you specify a dependent property value before its
parent property, an error or warning may occur.
H1 = hdlram('RAMType','Single port')
where
or
set(H1,'RAMType','Dual port')
25-5
25 System Objects
25-6
Process Data Using System Objects
Common Methods
All System objects support the following methods, each of which is described
in a method reference page associated with the particular object. In cases
where a method is not applicable to a particular object, calling that method
has no effect on the object.
25-7
25 System Objects
Method Description
step Processes data using the algorithm defined by the
object. As part of this processing, it initializes needed
resources, returns outputs, and updates the object
states. After you call the step method, you cannot
change any input specifications (i.e., dimensions, data
type, complexity). During execution, you can change
only tunable properties. The step method returns
regular MATLAB variables.
Example: Y = step(H,X)
release Releases any special resources allocated by the object,
such as file handles and device drivers, and unlocks
the object. For System objects, use the release
method instead of a destructor. See “Understand
System object Modes” on page 25-10.
clone Creates another object with the same property values
isLocked Returns a logical value indicating whether the object
is locked. See “Understand System object Modes” on
page 25-10.
reset Resets the internal states of the object to the initial
values for that object
isDone Applies to source objects only. Returns a logical value
indicating whether the step method has reached
the end of the data file. If a particular object does
not have end-of-data capability, this method value
returns false.
isChangedProperty Returns true if the specified tunable property value
has changed since the last call to step.Example: flag
= isChangedProperty(obj,'propertyName')
info Returns a structure containing characteristic
information about the object. The fields of this
structure vary depending on the object. If a particular
object does not have characteristic information, the
structure is empty.
25-8
Process Data Using System Objects
Method Description
getNumInputs Returns the number of inputs (excluding the object
itself) expected by the step method. This number
varies for an object depending on whether any
properties enable additional inputs.
getNumOutputs Returns the number of outputs expected from the step
method. This number varies for an object depending
on whether any properties enable additional outputs.
getDiscreteState Returns the discrete states of the object in a structure.
If the object is unlocked (when the object is first
created and before you have run the step method
on it or after you have released the object), the
states are empty. If the object has no discrete states,
getDiscreteState returns an empty structure.
These advantages make System objects particularly well suited for processing
streaming data, where segments of a continuous data stream are processed
iteratively. This ability to process streaming data provides the advantage of
not having to hold large amounts of data in memory. Use of streaming data
also allows you to use simplified programs that use loops efficiently.
25-9
25 System Objects
The object initializes and locks when it begins processing data. The typical
way in which an object becomes locked is when the step method is called on
that object. To determine if an object is locked, use the isLocked method. To
unlock an object, use the release method. When the object is locked, you
cannot change any of the following:
Several System objects do not allow changing the complexity of inputs from
real to complex. You can, however, change the input complexity from complex
to real without unlocking the object.
These restrictions allow the object to maintain states and allocate memory
appropriately.
25-10
Tuning System object™ Properties in MATLAB®
help hdlram.RAMType
where
Note Unless otherwise specified, System object properties are not tunable.
For objects that do not support variable-size input, if you change the input
dimensions while the object is in locked mode, the object produces a warning
and unlocks. The object then reinitializes the next time you call the step
method. See the object’s reference page for more information. You can change
the value of a tunable property and the input size without a warning or error
being produced. For all other changes at runtime, an error occurs.
25-11
25 System Objects
To view examples, go to the Help contents for the associated product. Under
Examples, select MATLAB Examples.
25-12
Use System Objects in MATLAB® Code Generation
The following example, which uses System objects, shows the key factors to
consider, such as passing property values and using extrinsic functions, when
you make MATLAB code suitable for code generation.
function lmssystemidentification
% LMSSYSTEMIDENTIFICATION System identification using
% LMS adaptive filter
%#codegen
25-13
25 System Objects
if isempty(hlms)
hfilt = dsp.DigitalFilter(...
'TransferFunction', 'FIR (all zeros)', ...
'Numerator', fir1(10, .25));
end
coder.extrinsic('stem');
For a detailed code generation example, see “Generate Code for MATLAB
Handle Classes and System Objects” in the MATLAB Coder product
documentation.
25-14
Use System Objects in MATLAB® Code Generation
The following usage rules and limitations apply to using System objects in
code generated from MATLAB.
• Do not set System objects to become outputs from the MATLAB Function
block.
• Do not use the Save and Restore Simulation State as SimState option for
any System object in a MATLAB Function block.
• Do not pass a System object as an example input argument to a function
being compiled with codegen.
• Do not pass a System object to functions declared as extrinsic (functions
called in interpreted mode) using the coder.extrinsic function. System
objects returned from extrinsic functions and scope System objects that
automatically become extrinsic can be used as inputs to another extrinsic
function, but do not generate code.
25-15
25 System Objects
f = coder.MEXConfig;
f.GlobalSyncMethod='NoSync'
Methods
• Code generation support is available only for these System object methods:
- get
- getNumInputs
- getNumOutputs
25-16
Use System Objects in MATLAB® Code Generation
• Code generation support for using dot notation depends on whether the
System object is predefined in the software or is one that you defined.
- For System objects that are predefined in the software, you cannot use
dot notation to call methods.
- For System objects that you define, you can use dot notation or function
call notation, with the System object as first argument, to call methods.
25-17
25 System Objects
Note MATLAB Compiler software supports System objects for use inside
MATLAB functions. The compiler product does not support System objects
for use in MATLAB scripts.
25-18
Fixed-Point Designer for
Simulink Models
26
Getting Started
Product Description
Design and simulate fixed-point systems
Fixed-Point Designer enables the fixed-point capabilities of the Simulink
product family, letting you use those products to design, simulate, and
implement fixed-point control and signal processing algorithms.
Key Features
• Fixed-point modeling and simulation in Simulink, Stateflow, and other
Simulink products
• Bit-true, fixed-point arithmetic for code generated by Simulink C, HDL,
and PLC code generation products
• Automated advisors that convert models from floating- to fixed-point data
types
• Analysis tools for deriving ranges for all signals based on design information
• Data type tools that use range data to recommend word length and scaling
• Control of fixed-point data type and of scaling from 1- to 128-bit word sizes
• Customizable fixed-point operators and math functions for embedded
code generation
26-2
What You Need to Get Started
Installation
To determine if the Fixed-Point Designer software is installed on your system,
type
ver
at the MATLAB command line. When you enter this command, the MATLAB
Command Window displays information about the version of MATLAB
software you are running, including a list of installed add-on products and
their version numbers. Check the list to see if the Fixed-Point Designer
software appears.
If you experience installation difficulties and have Web access, look for
the installation and license information at the MathWorks Web site
(http://www.mathworks.com/support).
26-3
26 Getting Started
If you do not have the Fixed-Point Designer software, you can work with a
model containing Simulink blocks with fixed-point settings as follows:
Note If you use fi objects or embedded numeric data types in your model or
workspace, you might introduce fixed-point data types into your model. You
can set fipref to prevent the checkout of a Fixed-Point Designer license.
26-4
Physical Quantities and Measurement Scales
Introduction
The decision to use fixed-point hardware is simply a choice to represent
numbers in a particular form. This representation often offers advantages
in terms of the power consumption, size, memory usage, speed, and cost of
the final product.
Well known standard scales like Celsius are very convenient for the exchange
of information. However, there are situations where it makes sense to create
and use unique nonstandard scales. These situations usually involve making
the most of a limited resource.
For example, nonstandard scales allow map makers to get the maximum
detail on a fixed size sheet of paper. A typical road atlas of the USA will show
each state on a two-page display. The scale of inches to miles will be unique
for most states. By using a large ratio of miles to inches, all of Texas can fit
on two pages. Using the same scale for Rhode Island would make poor use of
the page. Using a much smaller ratio of miles to inches would allow Rhode
Island to be shown with the maximum possible detail.
26-5
26 Getting Started
maximum detail. Similarly, the binary scale for a measurement should make
the most of the processor in order to get maximum precision.
26-6
Physical Quantities and Measurement Scales
100o C
Q1 Conversion 1
100
100o C
V Q2 Conversion 2
255
100o C
Q3 Conversion 3
200
Graphically, the conversion is a line with slope S, which must pass through
the origin. A line through the origin is called a purely linear conversion.
Restricting yourself to a purely linear conversion can be very wasteful and it
is often better to use the general equation of a line:
V = SQ + B.
By adding a bias term B, you can obtain greater precision when quantizing
to a limited number of bits.
V = SQ + B ± Error.
S S
− ≤ Error ≤ .
2 2
26-7
26 Getting Started
That is, the amount of quantization error is determined by both the number of
bits and by the scale. This scenario represents the best-case error. For other
rounding schemes, the error can be twice as large.
The temperature of the air flowing into the engine has a limited range. On a
typical engine, the radiator is designed to keep the block below the boiling
point of the cooling fluid. Assume a maximum of 225oF (380 K). As the
air flows through the intake manifold, it can be heated to this maximum
temperature. For a cold start in an extreme climate, the temperature can be
as low as -60oF (222 K). Therefore, using the absolute kelvin scale, the range
of interest is 222 K to 380 K.
380 K
V= Q.
7.5 bit
The quantized conversion and range of interest are shown in the following
figure.
26-8
Physical Quantities and Measurement Scales
Notice that there are 7.5 possible quantization values. This is because only
half of the first bit corresponds to temperatures (real-world values) greater
than zero.
26-9
26 Getting Started
The range of interest of the quantized conversion and the absolute value of
the quantized error are shown in the following figure.
26-10
Physical Quantities and Measurement Scales
The quantized conversion and range of interest are shown in the following
figure.
26-11
26 Getting Started
The range of interest of the quantized conversion and the absolute value of
the quantized error are shown in the following figure.
26-12
Physical Quantities and Measurement Scales
Clearly, the general linear scale gives much better precision than the purely
linear scale over the range of interest.
26-13
26 Getting Started
After making the decision to use fixed-point hardware, the next step is to
choose a method for implementing the dynamic system (for example, control
26-14
Why Use Fixed-Point Hardware?
26-15
26 Getting Started
You can configure the Fixed-Point Tool to automatically log the overflows,
saturations, and signal extremes of your simulations. You can also use it to
automate data typing and scaling decisions and to compare your fixed-point
implementations against idealized, floating-point benchmarks.
You can use the Fixed-Point Designer software with the Simulink
Coder product to automatically generate efficient, integer-only C code
representations of your designs. You can use this C code in a production
target or for rapid prototyping. In addition, you can use the Fixed-Point
Designer software with the Embedded Coder product to generate real-time
C code for use on an integer production, embedded target. You can also use
Fixed-Point Designer with HDL Coder to generate portable, synthesizable
VHDL and Verilog code from Simulink models and Stateflow charts.
26-16
Developing and Testing Fixed-Point Systems
26-17
26 Getting Started
26-18
Developing and Testing Fixed-Point Systems
Using the MATLAB, Simulink, and Fixed-Point Designer software, you follow
these steps of the development cycle:
1 Model the system (plant or signal source) within the Simulink software
using double-precision numbers. Typically, the model will contain
nonlinear elements.
When you have met the design requirements, you can use the model as a
specification for creating production code using the Simulink Coder product or
generating HDL code using the HDL Coder product.
The Fixed-Point Designer software helps expedite the design cycle by allowing
you to simulate the effects of various fixed-point controller and digital filter
structures.
26-19
26 Getting Started
The software supports all scaling choices including pure integer, binary point,
and slope bias. For slope bias scaling, it does not support complex fixed-point
types that have non-zero bias or non-trivial slope.
The save data type support extends to signals, parameters, and states.
26-20
Configure Blocks with Fixed-Point Output
26-21
26 Getting Started
In this section...
“Specify the Output Data Type and Scaling” on page 26-22
“Specify Fixed-Point Data Types with the Data Type Assistant” on page
26-24
“Rounding” on page 26-27
“Overflow Handling” on page 26-28
“Lock the Output Data Type Setting” on page 26-28
“Real-World Values Versus Stored Integer Values” on page 26-28
To configure blocks with Fixed-Point Designer data types, specify the data
type parameter on a block dialog box as an expression that evaluates to a
data type. Alternatively, you can use an assistant that simplifies the task of
entering data type expressions (see “Specify Fixed-Point Data Types with
26-22
Configure Blocks with Fixed-Point Output
the Data Type Assistant” on page 26-24). The sections that follow describe
varieties of fixed-point and floating-point data types, and the corresponding
functions that you use to specify them.
Integers
To specify unsigned and signed integers, use the uint and sint functions,
respectively.
For example, to configure a 16-bit unsigned integer via the block dialog box,
specify the Output data type parameter as uint(16). To configure a 16-bit
signed integer, specify the Output data type parameter as sint(16).
For integer data types, the default binary point is assumed to lie to the right
of all bits.
Fractional Numbers
To specify unsigned and signed fractional numbers, use the ufrac and sfrac
functions, respectively.
Both unsigned and signed fractional data types support guard bits, which
act to guard against overflow. For example, sfrac(16,4) specifies a 16-bit
signed fractional number with 4 guard bits. The guard bits lie to the left
of the default binary point.
26-23
26 Getting Started
Note Alternatively, you can use the fixdt function to create integer,
fractional, and generalized fixed-point objects. The fixdt function also allows
you to specify scaling for fixed-point data types.
Floating-Point Numbers
The Fixed-Point Designer software supports single-precision and
double-precision floating-point numbers as defined by the IEEE® Standard
754-1985 for Binary Floating-Point Arithmetic. You can specify floating-point
numbers with the Simulink float function.
You can use the Data Type Assistant to specify a fixed-point data type.
When you select Fixed point in the Mode field, the assistant displays fields
26-24
Configure Blocks with Fixed-Point Output
Signedness
Select whether you want the fixed-point data to be Signed or Unsigned.
Signed data can represent positive and negative quantities. Unsigned data
represents positive values only.
26-25
26 Getting Started
Word length
Specify the size (in bits) of the word that will hold the quantized integer.
Large word sizes represent large quantities with greater precision than small
word sizes. Fixed-point word sizes up to 128 bits are supported for simulation.
Scaling
Specify the method for scaling your fixed-point data to avoid overflow
conditions and minimize quantization errors. You can select the following
scaling modes:
Scaling Description
Mode
Binary If you select this mode, the assistant displays the Fraction length field,
point specifying the binary point location.
Binary points can be positive or negative integers. A positive integer moves the
binary point left of the rightmost bit by that amount. For example, an entry of 2
sets the binary point in front of the second bit from the right. A negative integer
moves the binary point further right of the rightmost bit by that amount, as
in this example:
26-26
Configure Blocks with Fixed-Point Output
Scaling Description
Mode
See “Slope and Bias Scaling” on page 27-8 for more information.
Best If you select this mode, the block scales a constant vector or matrix such that
precision the precision of its elements is maximized. This mode is available only for
particular blocks.
See “Constant Scaling for Best Precision” on page 27-15 for more information.
Rounding
You specify how fixed-point numbers are rounded with the Integer rounding
mode parameter. The following rounding modes are supported:
26-27
26 Getting Started
For more information about each of these rounding modes, see “Rounding”
on page 28-4.
Overflow Handling
To control how overflow conditions are handled for fixed-point operations, use
the Saturate on integer overflow check box.
If this box is not selected, overflows wrap to the appropriate value that is
representable by the data type. For example, the number 130 does not fit in a
signed 8-bit integer, and would wrap to -126.
When locked, the Fixed-Point Tool and automatic scaling script autofixexp
do not change the output data type setting. Otherwise, the Fixed-Point Tool
and autofixexp script are free to adjust the output data type setting.
26-28
Configure Blocks with Fixed-Point Output
The possible values are Real World Value (RWV) and Stored Integer
(SI).
26-29
26 Getting Started
Note S-functions and the Stateflow Chart block do not support fixed-point
parameters.
You can specify a fixed-point parameter value either directly by setting the
value of the parameter to an expression that evaluates to a fi object, or
indirectly by setting the value of the parameter to an expression that refers to
a fixed-point Simulink.Parameter object.
In this section...
“Specify Fixed-Point Values Directly” on page 26-30
“Specify Fixed-Point Values Via Parameter Objects” on page 26-31
fi(3.3,1,8,3)
as the Constant value parameter for the Constant block specifies a signed
fixed-point value of 3.3, with a word length of 8 bits and a fraction length
of 3 bits.
26-30
Configure Blocks with Fixed-Point Parameters
For example, suppose you want to create a fixed-point constant in your model.
You could do this using a fixed-point parameter object and a Constant block
as follows:
my_fixpt_param = Simulink.Parameter
26-31
26 Getting Started
my_fixpt_param.Value = fi(3.3,true,8,3)
Alternatively, you can set the parameter object’s Value and DataType
properties separately. In this case, specify the relevant fixed-point data
type using a Simulink.AliasType object, a Simulink.NumericType object,
or a fixdt expression. For example, the following commands independently
set the parameter object’s value and data type, using a fixdt expression as
the DataType string:
my_fixpt_param.Value = 3.3;
my_fixpt_param.DataType = 'fixdt(true,8,2^-3,0)'
26-32
Configure Blocks with Fixed-Point Parameters
26-33
26 Getting Started
For example, you can use the following code to create a structure in the
MATLAB workspace with a fi object in the values field. You can then use
the From Workspace block to bring the data into a Simulink model.
a = fi([sin(0:10)' sin(10:-1:0)'])
26-34
Pass Fixed-Point Data Between Simulink® Models and MATLAB®
a =
0 -0.5440
0.8415 0.4121
0.9093 0.9893
0.1411 0.6570
-0.7568 -0.2794
-0.9589 -0.9589
-0.2794 -0.7568
0.6570 0.1411
0.9893 0.9093
0.4121 0.8415
-0.5440 0
s =
s.signals.dimensions = 2
s =
s.time = [0:10]'
s =
The From Workspace block in the following model has the fi structure s in
the Data parameter. In the model, the following parameters in the Solver
pane of the Configuration Parameters dialog box have the indicated settings:
26-35
26 Getting Started
The To Workspace block writes the result of the simulation to the MATLAB
workspace as a fi structure.
simout.signals.values
ans =
26-36
Pass Fixed-Point Data Between Simulink® Models and MATLAB®
0 -8.7041
13.4634 6.5938
14.5488 15.8296
2.2578 10.5117
-12.1089 -4.4707
-15.3428 -15.3428
-4.4707 -12.1089
10.5117 2.2578
15.8296 14.5488
6.5938 13.4634
-8.7041 0
When you log signals from a referenced model or Stateflow chart in your
model, the word lengths of fi objects may be larger than you expect. The word
lengths of fixed-point signals in referenced models and Stateflow charts are
logged as the next larger data storage container size.
26-37
26 Getting Started
as fi objects. For more information about the API, refer to “Access Block Data
During Simulation”.
26-38
Cast from Doubles to Fixed Point
• Data types
• Scaling
• Rounding
• Logging minimum and maximum simulation values to the workspace
• Overflow handling
fxpdemo_dbl2fix
26-39
26 Getting Started
The sections that follow describe the model and its simulation results.
Block Descriptions
In this example, you configure the Signal Generator block to output a sine
wave signal with an amplitude defined on the interval [-5 5]. The Signal
Generator block always outputs double-precision numbers.
26-40
Cast from Doubles to Fixed Point
The Data Type Conversion (FixPt-to-Dbl) block converts one of the Fixed-Point
Designer data types into a Simulink data type. In this example, it outputs
double-precision numbers.
Simulations
The following sections describe how to simulate the model using
binary-point-only scaling and [Slope Bias] scaling.
Binary-Point-Only Scaling
When using binary-point-only scaling, your goal is to find the optimal
power-of-two exponent E, as defined in “Scaling” on page 27-6. For this scaling
mode, the fractional slope F is 1 and there is no bias.
1 Configure the Signal Generator block to output a sine wave signal with an
amplitude defined on the interval [-5 5].
a Double-click the Signal Generator block to open the Block Parameters
dialog.
b Set the Wave form parameter to sine.
d Click OK.
26-41
26 Getting Started
26-42
Cast from Doubles to Fixed Point
If you want to span the complete range of the input signal with 5 bits using
binary-point-only scaling, then your only option is to sacrifice precision.
Hence, the output scaling is 2^-1, which puts the binary point one place to
the left of the rightmost bit. This scaling gives a maximum value of 0111.1 =
7.5, a minimum value of 1000.0 = -8.0, and a precision of (1/2)1 = 0.5.
To simulate using a precision of 0.5, set the Output data type parameter of
the Data Type Conversion (Dbl-to-FixPt) block to fixdt(1,5,1) and rerun
the simulation.
To arrive at a value for the slope, you begin by assuming a fixed power-of-two
exponent of -2. To find the fractional slope, you divide the maximum value of
the sine wave by the maximum value of the scaled 5-bit number. The result
is 5.00/3.75 = 1.3333. The slope (and precision) is 1.3333.(0.25) = 0.3333.
You specify the [Slope Bias] scaling as [0.3333 0] by entering the expression
fixdt(1,5,0.3333,0) as the value of the Output data type parameter.
1 Configure the Signal Generator block to output a sine wave signal with an
amplitude defined on the interval [-5 5].
a Double-click the Signal Generator block to open the Block Parameters
dialog.
b Set the Wave form parameter to sine.
26-43
26 Getting Started
d Click OK.
26-44
Cast from Doubles to Fixed Point
You do not need to find the slope using this method. You need only the range
of the data you are simulating and the size of the fixed-point word used in the
simulation. You can achieve reasonable simulation results by selecting your
scaling based on the formula
where
26-45
26 Getting Started
26-46
27
With the Fixed-Point Designer software, you can explore the relationship
between data types, range, precision, and quantization error in the modeling
of dynamic digital systems. With the Simulink Coder product, you can
generate production code based on that model. With HDL Coder, you can
generate portable, synthesizable VHDL and Verilog code from Simulink
models and Stateflow charts.
27-2
Fixed-Point Numbers
Fixed-Point Numbers
In this section...
“Fixed-Point Numbers” on page 27-3
“Signed Fixed-Point Numbers” on page 27-4
“Binary Point Interpretation” on page 27-4
“Scaling” on page 27-6
“Quantization” on page 27-8
“Range and Precision” on page 27-10
Fixed-Point Numbers
Fixed-point numbers and their data types are characterized by their word
size in bits, binary point, and whether they are signed or unsigned. The
Fixed-Point Designer software supports integers and fixed-point numbers.
The main difference among these data types is their binary point.
where
27-3
27 Data Types and Scaling
• The most significant bit (MSB) is the leftmost bit, and is represented by
location bws1
• The least significant bit (LSB) is the rightmost bit, and is represented by
location b0
• The binary point is shown four places to the left of the LSB
FractionLength defines the scaling of the stored integer value. The word
length limits the values that the stored integer can take, but it does not limit
the values FractionLength can take. The software does not restrict the value
27-4
Fixed-Point Numbers
bbb. = bbb. × 20
bb.b = bbb. × 2−1
b.bb = bbb. × 2−2
.bbb = bbb. × 2−3
If the exponent were greater than 0 or less than -3, then the representation
would involve lots of zeros.
bbb00000. = bbb. × 25
bbb00. = bbb. × 22
.00bbb = bbb. × 2−5
.00000bbb = bbb. × 2−8
These extra zeros never change to ones, however, so they don’t show up in
the hardware. Furthermore, unlike floating-point exponents, a fixed-point
exponent never shows up in the hardware, so fixed-point exponents are not
limited by a finite number of bits.
Consider a signed value with a word length of 8, a fraction length of 10, and
a stored integer value of 5 (binary value 00000101). The real-word value is
calculated using the formula
RealWorldValue = StoredInteger * 2 ^ -FractionLength. In this case,
RealWorldValue = 5 * 2 ^ -10 = 0.0048828125. Because the fraction
length is 2 bits longer than the word length, the binary value of the stored
integer is x.xx00000101 , where x is a placeholder for implicit zeros.
0.0000000101 (binary) is equivalent to 0.0048828125 (decimal). For an
example using a fi object, see “Create a fi Object With Fraction Length
Greater Than Word Length”.
27-5
27 Data Types and Scaling
Scaling
The dynamic range of fixed-point numbers is much less than floating-point
numbers with equivalent word sizes. To avoid overflow conditions and
minimize quantization errors, fixed-point numbers must be scaled.
With the Fixed-Point Designer software, you can select a fixed-point data type
whose scaling is defined by its binary point, or you can select an arbitrary
linear scaling that suits your needs. This section presents the scaling choices
available for fixed-point data types.
You can represent a fixed-point number by a general slope and bias encoding
scheme
~
V ≈ V = SQ + B,
where
~
• V is the approximate real-world value.
• Q, the stored value, is an integer that encodes V.
• S F 2 E is the slope.
• B is the bias.
Note S and B are constants and do not show up in the computer hardware
directly. Only the quantization value Q is stored in computer memory.
27-6
Fixed-Point Numbers
The scaling modes available to you within this encoding scheme are described
in the sections that follow. For detailed information about how the supported
scaling modes effect fixed-point operations, refer to “Recommendations for
Arithmetic and Scaling” on page 28-33.
Binary-Point-Only Scaling
Binary-point-only or power-of-two scaling involves moving the binary point
within the fixed-point word. The advantage of this scaling mode is to minimize
the number of processor arithmetic operations.
With binary-point-only scaling, the components of the general slope and bias
formula have the following values:
• F=1
• S F2E 2E
• B0
Integers are a special case of fixed-point data types. Integers have a trivial
scaling with slope 1 and bias 0, or equivalently with fraction length 0. Specify
integers as
27-7
27 Data Types and Scaling
Unspecified Scaling
Specify fixed-point data types with an unspecified scaling as
• fixdt(Signed, WordLength)
Quantization
The quantization Q of a real-world value V is represented by a weighted sum
of bits. Within the context of the general slope and bias encoding scheme, the
value of an unsigned fixed-point quantity is given by
~ ⎡ ws−1 ⎤
V = S. ⎢ ∑ bi 2i ⎥ + B,
⎢⎣ i=0 ⎥⎦
~ ⎡ ws−2 ⎤
V = S. ⎢ −bws−1 2ws−1 + ∑ bi 2i ⎥ + B,
⎢⎣ i =0 ⎥⎦
where
27-8
Fixed-Point Numbers
Fixed-Point Format
Formats for 8-bit signed and unsigned fixed-point values are shown in the
following figure.
Note that you cannot discern whether these numbers are signed or unsigned
data types merely by inspection since this information is not explicitly
encoded within the word.
The binary number 0011.0101 yields the same value for the unsigned and
two’s complement representation because the MSB = 0. Setting B = 0 and
using the appropriate weights, bit multipliers, and scaling, the value is
⎡ ws−1 ⎤
( )
~
V = F 2 E Q = 2 E ⎢ ∑ bi 2i ⎥
⎢⎣ i=0 ⎥⎦
(
= 2−4 0 × 27 + 0 × 26 + 1 × 25 + 1 × 24 + 0 × 23 + 1 × 22 + 0 × 21 + 1 × 20 )
= 3.3125.
Conversely, the binary number 1011.0101 yields different values for the
unsigned and two’s complement representation since the MSB = 1.
Setting B = 0 and using the appropriate weights, bit multipliers, and scaling,
the unsigned value is
27-9
27 Data Types and Scaling
⎡ ws−1 ⎤
( )
~
V = F 2 E Q = 2 E ⎢ ∑ bi 2i ⎥
⎢⎣ i=0 ⎥⎦
(
= 2−4 1 × 27 + 0 × 26 + 1 × 25 + 1 × 24 + 0 × 23 + 1 × 22 + 0 × 21 + 1 × 20 )
= 11.3125,
⎡ ws−2 ⎤
( )
~
V = F 2 E Q = 2 E ⎢ −bws−1 2ws−1 + ∑ bi 2i ⎥
⎢⎣ i =0 ⎥⎦
(
= 2−4 −1 × 27 + 0 × 26 + 1 × 25 + 1 × 24 + 0 × 23 + 1 × 22 + 0 × 21 + 1 × 20 )
= −4.6875.
Range
The following figure illustrates the range of representable numbers for an
unsigned fixed-point number of size ws, scaling S, and bias B.
27-10
Fixed-Point Numbers
For both the signed and unsigned fixed-point numbers of any data type, the
number of different bit patterns is 2ws.
For example, if the fixed-point data type is an integer with scaling defined
as S 1 and B = 0, then the maximum unsigned value is 2ws1 , because
zero must be represented. In two’s complement, negative numbers must
be represented as well as zero, so the maximum value is 2ws1 1 .
Additionally, since there is only one representation for zero, there must be
an unequal number of positive and negative numbers. This means there is a
representation for 2ws1 but not for 2ws1 .
Precision
The precision of a data type is given by the slope. In this usage, precision
means the difference between neighboring representable values.
27-11
27 Data Types and Scaling
Default
Scaling
Name Data Type Low Limit High Limit (~Precision)
Unsigned fixdt(0,ws,0) 0 1
Integer 2ws 1
Signed fixdt(1,ws,0) 1
Integer 2ws1 2ws1 1
Unsigned fixdt(0,ws,fl) 0
Binary (2ws 1)2 fl 2 fl
Point
Signed fixdt(1,ws,fl)
Binary 2ws1 fl (2ws1 1)2 fl 2 fl
Point
Unsigned fixdt(0,ws,s,b) b s
Slope s(2ws 1) b
Bias
Signed fixdt(1,ws,s,b) s
Slope s(2ws1 ) b s(2ws1 1) b
Bias
27-12
Fixed-Point Numbers
Range of Signed
Values (Low, Range of Unsigned
Scaling Precision High) Values (Low, High)
21 2.0 -256, 254 0, 510
2 0
1.0 -128, 127 0, 255
2 -1
0.5 -64, 63.5 0, 127.5
2 -2
0.25 -32, 31.75 0, 63.75
2-3 0.125 -16, 15.875 0, 31.875
2 -4
0.0625 -8, 7.9375 0, 15.9375
2 -5
0.03125 -4, 3.96875 0, 7.96875
2 -6
0.015625 -2, 1.984375 0, 3.984375
2-7 0.0078125 -1, 0.9921875 0, 1.9921875
2-8 0.00390625 -0.5, 0.49609375 0, 0.99609375
27-13
27 Data Types and Scaling
27-14
Fixed-Point Numbers in Simulink
• Constant
• Discrete FIR Filter
• Gain
• Relay
• Repeating Sequence Stair
This scaling mode is based on binary-point-only scaling. Using this mode, you
can scale a constant vector or matrix such that a common binary point is
found based on the best precision for the largest value in the vector or matrix.
Constant scaling for best precision is available only for fixed-point data types
with unspecified scaling. All other fixed-point data types use their specified
scaling. You can use the Data Type Assistant (see “Specify Data Types
Using Data Type Assistant”) on a block dialog box to enable the best precision
scaling mode.
1 On a block dialog box, click the Show data type assistant button
2 In the Data Type Assistant, and from the Mode list, select Fixed point.
27-15
27 Data Types and Scaling
27-16
Fixed-Point Numbers in Simulink
To understand how you might use this scaling mode, consider a 3-by-3 matrix
of doubles, M, defined as
Now suppose you specify M as the value of the Gain parameter for a Gain
block. The results of specifying your own scaling versus using the constant
scaling mode are described here:
• Specified Scaling
Suppose the matrix elements are converted to a signed, 10-bit generalized
fixed-point data type with binary-point-only scaling of 2-7 (that is, the
binary point is located seven places to the left of the right most bit). With
this data format, M becomes
0 0 0
3.1250e-002 0 0
3.3594e-001 3.1250e-002 0
Note that many of the matrix elements are zero, and for the nonzero
entries, the scaled values differ from the original values. This is because a
double is converted to a binary word of fixed size and limited precision for
each element. The larger and more precise the conversion data type, the
more closely the scaled values match the original values.
• Constant Scaling for Best Precision
If M is scaled based on its largest matrix value, you obtain
2.9297e-003 0 0
3.3203e-002 2.9297e-003 0
3.3301e-001 3.3203e-002 2.9297e-003
27-17
27 Data Types and Scaling
You can represent a fixed-point number using the fixed-point scaling equation
~
V ≈ V = SQ + B,
where
• S F 2 E is the slope.
• F is the slope adjustment factor.
• E is the fixed power-of-two exponent.
• Q is the stored integer.
• B is the bias.
The following table provides a key for various symbols that appear in
Simulink products to indicate the data type and scaling of a fixed-point value.
27-18
Fixed-Point Numbers in Simulink
27-19
27 Data Types and Scaling
Scaled Doubles
27-20
Fixed-Point Numbers in Simulink
Consider an example where you are storing 0.75001 degrees Celsius in a data
type sfix16_En13. For this data type:
• The bias, B 0 .
~
Using the scaling equation V ≈ V = SQ + B, where V is the real-world value
and Q is the stored value.
• B0.
~
• V SQ 213 Q 0.75001 .
Because the storage container of the data type sfix16_En13 is 16 bits, the
stored integer Q can only be represented as an integer within these 16 bits, so
the ideal value of Q is quantized to 6144 causing precision loss.
If you override the data type sfix16_En13 with Double, the data type changes
to Double and you lose the information about the scaling. The stored-value
equals the real-world value 0.75001.
If you override the data type sfix16_En13 with Scaled Double, the data type
changes to flts16_En13. The scaling is still given by _En13 and is identical
to that of the original data type. The only difference is the storage container
used to hold the stored value which is now double so the stored-value is
6144.08192. This example shows one advantage of using scaled doubles: the
virtual elimination of quantization errors.
27-21
27 Data Types and Scaling
27-22
Fixed-Point Numbers in Simulink
In this model:
27-23
27 Data Types and Scaling
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_scaled_double
3 In the Fixed-Point Tool, set the Data type override parameter to Use
local settings and click Apply.
The simulation runs and the Display block displays 4.125 as the output
value of the Constant block. The Stored Integer Display block displays
0100 0010, which is the binary equivalent of the stored integer value.
Precision loss occurs because the output data type, fixdt(1,8,4), cannot
represent the output value 4.1 exactly.
5 In the Fixed-Point Tool, set the Data type override parameter to Scaled
double and the Data type override applies to parameter to All
numeric types. Then click Apply and rerun the simulation.
Note You cannot use a Data type override setting of Double because the
Bitwise Operator block does not support floating-point data types.
The simulation runs and this time the Display block correctly displays 4.1
as the output value of the Constant block. The Stored Integer Display
block displays 65, which is the binary equivalent of the stored integer
value. Because the model uses scaled doubles to override the data type
fixdt(1,8,4), the compiled output data type changes to flts8_En4, which
is the scaled doubles equivalent of fixdt(1,8,4). No precision loss occurs
because the scaled doubles retain the information about the specified data
type and scaling, and they use a double to hold the stored value.
27-24
Fixed-Point Numbers in Simulink
1 From the Simulink Display menu, select Signals and Ports, and then
select Port Data Types.
The port display for fixed-point signals consists of three parts: the data
type, the number of bits, and the scaling. These three parts reflect the block
Output data type parameter value or the data type and scaling that is
inherited from the driving block or through back propagation.
27-25
27 Data Types and Scaling
In the model, the data type displayed with the In1 block indicates that the
output data type name is sfix16_Sp2_B10. This corresponds to fixdt(1,
16, 0.2, 10) which is a signed 16 bit fixed-point number with slope 0.2 and
bias 10.0. The data type displayed with the In2 block indicates that the
output data type name is sfix16_En6. This corresponds to fixdt(1, 16, 6)
which is a signed 16 bit fixed-point number with fraction length of 6.
27-26
Floating-Point Numbers
Floating-Point Numbers
In this section...
“Floating-Point Numbers” on page 27-27
“Scientific Notation” on page 27-28
“The IEEE Format” on page 27-28
“Range and Precision” on page 27-30
“Exceptional Arithmetic” on page 27-33
Floating-Point Numbers
Fixed-point numbers are limited in that they cannot simultaneously represent
very large or very small numbers using a reasonable word size. This limitation
can be overcome by using scientific notation. With scientific notation, you can
dynamically place the binary point at a convenient location and use powers of
the binary to keep track of that location. Thus, you can represent a range of
very large and very small numbers with only a few digits.
27-27
27 Data Types and Scaling
Scientific Notation
A direct analogy exists between scientific notation and radix point notation.
For example, scientific notation using five decimal digits for the fraction
would take the form
Radix point notation using five bits for the fraction is the same except for
the number base
For fixed-point numbers, the exponent is fixed but there is no reason why the
binary point must be contiguous with the fraction. For more information, see
“Binary Point Interpretation” on page 27-4.
27-28
Floating-Point Numbers
Single-Precision Format
The IEEE single-precision floating-point format is a 32-bit word divided into a
1-bit sign indicator s, an 8-bit biased exponent e, and a 23-bit fraction f. For
more information, see “The Sign Bit” on page 27-28, “The Exponent Field”
on page 27-29, and “The Fraction Field” on page 27-29. A representation of
this format is given below.
The relationship between this format and the representation of real numbers
is given by
27-29
27 Data Types and Scaling
Double-Precision Format
The IEEE double-precision floating-point format is a 64-bit word divided into
a 1-bit sign indicator s, an 11-bit biased exponent e, and a 52-bit fraction f.For
more information, see “The Sign Bit” on page 27-28, “The Exponent Field” on
page 27-29, and “The Fraction Field” on page 27-29. A representation of this
format is shown in the following figure.
The relationship between this format and the representation of real numbers
is given by
27-30
Floating-Point Numbers
Range
The range of representable numbers for an IEEE floating-point number with f
bits allocated for the fraction, e bits allocated for the exponent, and the bias of
e given by bias 2( e1) 1 is given below.
where
• Positive numbers less than 2(1bias ) and negative numbers less than
2(1bias) are either underflows or denormalized numbers.
Note You can use the MATLAB commands realmin and realmax to
determine the dynamic range of double-precision floating-point values for
your computer.
27-31
27 Data Types and Scaling
Precision
Because of a finite word size, a floating-point number is only an approximation
of the “true” value. Therefore, it is important to have an understanding of the
precision (or accuracy) of a floating-point result. In general, a value v with an
accuracy q is specified by v q . For IEEE floating-point numbers,
v = (–1)s(2e–bias)(1.f)
and
q = 2–f×2e–bias
Thus, the precision is associated with the number of bits in the fraction field.
27-32
Floating-Point Numbers
Exceptional Arithmetic
In addition to specifying a floating-point format, the IEEE Standard 754
specifies practices and procedures so that predictable results are produced
independently of the hardware platform. Specifically, denormalized numbers,
Inf, and NaN are defined to deal with exceptional arithmetic (underflow and
overflow).
Denormalized Numbers
Denormalized numbers are used to handle cases of exponent underflow. When
the exponent of the result is too small (i.e., a negative exponent with too large
a magnitude), the result is denormalized by right-shifting the fraction and
leaving the exponent at its minimum value. The use of denormalized numbers
is also referred to as gradual underflow. Without denormalized numbers, the
gap between the smallest representable nonzero number and zero is much
wider than the gap between the smallest representable nonzero number and
the next larger number. Gradual underflow fills that gap and reduces the
impact of exponent underflow to a level comparable with roundoff among the
normalized numbers. Thus, denormalized numbers provide extended range
for small numbers at the expense of precision.
Inf
Arithmetic involving Inf (infinity) is treated as the limiting case of real
arithmetic, with infinite values defined as those outside the range of
representable numbers, or –∞ ≤ (representable numbers) < ∞. With the
exception of the special cases discussed below (NaN), any arithmetic operation
involving Inf yields Inf. Inf is represented by the largest biased exponent
allowed by the format and a fraction of zero.
NaN
A NaN (not-a-number) is a symbolic entity encoded in floating-point format.
There are two types of NaN: signaling and quiet. A signaling NaN signals an
27-33
27 Data Types and Scaling
Both types of NaN are represented by the largest biased exponent allowed by
the format and a fraction that is nonzero. The bit pattern for a quiet NaN is
given by 0.f where the most significant number in f must be a one, while the bit
pattern for a signaling NaN is given by 0.f where the most significant number
in f must be zero and at least one of the remaining numbers must be nonzero.
27-34
28
Arithmetic Operations
• Overflow
Adding two sufficiently large negative or positive values can produce a
result that does not fit into the representation. This will have an adverse
effect on the control system.
• Quantization
Fixed-point values are rounded. Therefore, the output signal to the
plant and the input signal to the control system do not have the same
characteristics as the ideal discrete-time signal.
• Computational noise
The accumulated errors that result from the rounding of individual terms
within the realization introduce noise into the control signal.
• Limit cycles
In the ideal system, the output of a stable transfer function (digital filter)
approaches some constant for a constant input. With quantization, limit
cycles occur where the output oscillates between two values in steady state.
28-2
Precision
Precision
In this section...
“Limitations on Precision” on page 28-3
“Rounding” on page 28-4
“Choose a Rounding Mode” on page 28-4
“Rounding Modes for Fixed-Point Simulink Blocks” on page 28-5
“Rounding Mode: Ceiling” on page 28-7
“Rounding Mode: Convergent” on page 28-8
“Rounding Mode: Floor” on page 28-10
“Rounding Mode: Nearest” on page 28-11
“Rounding Mode: Round” on page 28-12
“Rounding Mode: Simplest” on page 28-14
“Rounding Mode: Zero” on page 28-17
“Pad with Trailing Zeros” on page 28-19
“Limitations on Precision and Errors” on page 28-20
“Maximize Precision” on page 28-21
“Net Slope and Net Bias Precision” on page 28-21
“Detect Net Slope and Bias Precision Issues” on page 28-24
“Detect Fixed-Point Constant Precision Loss” on page 28-25
Limitations on Precision
Computer words consist of a finite numbers of bits. This means that the
binary encoding of variables is only an approximation of an arbitrarily precise
real-world value. Therefore, the limitations of the binary representation
automatically introduce limitations on the precision of the value. For a
general discussion of range and precision, refer to “Range and Precision” on
page 27-10.
The precision of a fixed-point word depends on the word size and binary point
location. Extending the precision of a word can always be accomplished with
28-3
28 Arithmetic Operations
more bits, but you face practical limitations with this approach. Instead, you
must carefully select the data type, word size, and scaling such that numbers
are accurately represented. Rounding and padding with trailing zeros are
typical methods implemented on processors to deal with the precision of
binary words.
Rounding
The result of any operation on a fixed-point number is typically stored in a
register that is longer than the number’s original format. When the result is
put back into the original format, the extra bits must be disposed of. That is,
the result must be rounded. Rounding involves going from high precision to
lower precision and produces quantization errors and computational noise.
For more information on when to use each rounding mode, see “Rounding
Methods” on page 1-10 in the Fixed-Point Designer User’s Guide.
28-4
Precision
The following table illustrates the differences between these rounding modes:
28-5
28 Arithmetic Operations
28-6
Precision
In the MATLAB software, you can round to ceiling using the ceil function.
Rounding toward ceiling is shown in the following figure.
28-7
28 Arithmetic Operations
28-8
Precision
In the MATLAB software, you can perform convergent rounding using the
convergent function. Convergent rounding is shown in the following figure.
28-9
28 Arithmetic Operations
In the MATLAB software, you can round to floor using the floor function.
Rounding toward floor is shown in the following figure.
28-10
Precision
In the Fixed-Point Designer software, you can round to nearest using the
nearest function. Rounding toward nearest is shown in the following figure.
28-11
28 Arithmetic Operations
As a result:
28-12
Precision
In the MATLAB software, you can perform this type of rounding using the
round function. The rounding mode Round is shown in the following figure.
28-13
28 Arithmetic Operations
In nearly all cases, the simplest rounding mode produces the most efficient
generated code. For a very specialized case of division that meets three specific
criteria, round to floor might be more efficient. These three criteria are:
For this case, set the rounding mode to floor and the Model Configuration
Parameters > Hardware Implementation > Production Hardware >
Signed integer division rounds to parameter to describe the rounding
behavior of your production target.
• When casting from one integer or fixed-point data type to another, the
simplest mode rounds toward floor.
• When casting from a floating-point data type to an integer or fixed-point
data type, the simplest mode rounds toward zero.
28-14
Precision
28-15
28 Arithmetic Operations
cannot rely on your production target to perform the rounding, because the
parameter is Undefined. Therefore, you need additional rounding code to
ensure rounding to zero behavior.
u1 S Q + B1
y= = 1 1
u2 S2 Q2 + B2
The simplest rounding mode performs the best rounding for each of these
operations, high-level and intermediate, to produce the most efficient code.
The rules used to select the appropriate rounding for intermediate arithmetic
operations are the same as those described in “Optimize Rounding for
High-Level Arithmetic Operations” on page 28-15. Again, this enables
28-16
Precision
mixed-mode rounding, with the most common case being round toward floor
used for additions, subtractions, and multiplies, and round toward zero used
for divides.
Remember that generating the most efficient code using the simplest
rounding mode requires you to set the Model Configuration Parameters >
Hardware Implementation > Production Hardware > Signed integer
division rounds to parameter to describe the rounding behavior of your
production target.
28-17
28 Arithmetic Operations
28-18
Precision
truncation are different for unsigned and two’s complement numbers. For
this reason, the ambiguous term “truncation” is not used in this guide, and
explicit rounding modes are used instead.
For example, suppose two numbers are subtracted from each other. First,
the exponents must be aligned, which typically involves a right shift of the
number with the smaller value. In performing this shift, significant digits
can “fall off” to the right. However, when the appropriate number of extra
bits is appended, the precision of the result is maximized. Consider two 8-bit
fixed-point numbers that are close in value and subtracted from each other:
1.0000000 × 2q
−0.1111111 × 2q
.
0.0000001 × 2q
28-19
28 Arithmetic Operations
If the top number is padded by two zeros and the bottom number is padded
with one zero, then the above equation becomes
1.000000000 × 2q
−0.111111110 × 2q
,
0.000000010 × 2q
V ≈ V = SQ + B = 2−2 Q + 32,
where V = 35.375.
The two closest approximations to the real-world value are Q = 13 and Q = 14:
S F2E
V − V = 0.125 = = .
2 2
For fixed-point values within the limited range, this represents the worst-case
error if round-to-nearest is used. If other rounding modes are used, the
worst-case error can be twice as large:
28-20
Precision
V − V < F 2 E.
Maximize Precision
Precision is limited by slope. To achieve maximum precision, you should
make the slope as small as possible while keeping the range adequately large.
The bias is adjusted in coordination with the slope.
Assume the maximum and minimum real-world values are given by max(V)
and min(V), respectively. These limits might be known based on physical
principles or engineering considerations. To maximize the precision, you must
decide upon a rounding scheme and whether overflows saturate or wrap.
To simplify matters, this example assumes the minimum real-world value
corresponds to the minimum encoded value, and the maximum real-world
value corresponds to the maximum encoded value. Using the encoding scheme
described in “Scaling” on page 27-6, these values are given by
max ( V ) = F 2 E ( max ( Q ) ) + B
min ( V ) = F 2 E ( min ( Q ) ) + B.
V ≈ V = SQ + B,
28-21
28 Arithmetic Operations
where:
Sa Qa + Ba = SbQb + Bb
or
SbQb ⎛ Bb − Ba ⎞
Qa = +⎜ ⎟,
Sa ⎝ Sa ⎠
where:
Sb
• Sa is the net slope.
Bb − Ba
• Sa is the net bias.
You can configure your model to alert you when fixed-point constant precision
issues occur. For more information, see “Detect Net Slope and Bias Precision
Issues” on page 28-24. The Fixed-Point Designer software provides the
following information:
28-22
Precision
This information warns you that the outputs from this block are not accurate.
If possible, change the data types in your model to fix the issue.
28-23
28 Arithmetic Operations
For example, the maximum value that int8 can represent is 127. The
precision for int8 is 1.0. An ideal value of 127.3 quantizes to 127 with
an absolute error of 0.3. Although the ideal value 127.3 is greater than
the maximum representable value for int8, the quantization error is small
relative to the precision of int8. Therefore the Fixed-Point Designer software
does not report an overflow. However, an ideal value of 128.1 does cause
an overflow because the quantization error is 1.1, which is larger than the
precision for int8.
28-24
Precision
• Input slope, SU = 1
• Output slope, SY = 1.000001
• Net slope, SU/SY = 1/1.000001
When you simulate the model, a net slope quantization error occurs.
1 For the Inport block, set the Output data type to int16.
28-25
28 Arithmetic Operations
2 For the Data Type Conversion block, set the Output data type to
fixdt(1,16, 1.000001, 0).
3 Set the Diagnostics > Type Conversion > Detect precision loss
configuration parameter to error.
28-26
Range
Range
In this section...
“Limitations on Range” on page 28-27
“What Are Saturation and Wrapping?” on page 28-28
“Saturation and Wrapping” on page 28-28
“Guard Bits” on page 28-31
“Determine the Range of Fixed-Point Numbers” on page 28-31
Limitations on Range
Limitations on the range of a fixed-point word occur for the same reason as
limitations on its precision. Namely, fixed-point words have limited size. For
a general discussion of range and precision, refer to “Range and Precision” on
page 27-10.
The Simulink software supports saturation and wrapping for all fixed-point
data types, while guard bits are supported only for fractional data types.
As shown in the following figure, you can select saturation or wrapping for
fixed-point Simulink blocks with the Saturate on integer overflow check
box. By setting Output data type to sfrac(36,4), you specify a 36–bit
signed fractional data type with 4 guard bits (total word size is 40 bits).
28-27
28 Arithmetic Operations
28-28
Range
all negative values are set to zero, which is the smallest number representable
by the data type. The saturation of overflows is shown in the following figure.
Overflows Saturate
8
4
Negative values Negative values
saturate to zero saturate to zero
0
0 0.4 0.8 1.2 1.6 2
Time
If overflows wrap, all negative values are set to the appropriate positive value.
The wrapping of overflows is shown in the following figure.
28-29
28 Arithmetic Operations
Overflows Wrap
8
2
Negative values Negative values
wrap to positive wrap to positive
values. values.
0
0 0.4 0.8 1.2 1.6 2
Time
Note For most control applications, saturation is the safer way of dealing
with fixed-point overflow. However, some processor architectures allow
automatic saturation by hardware. If hardware saturation is not available,
then extra software is required, resulting in larger, slower programs. This
cost is justified in some designs—perhaps for safety reasons. Other designs
accept wrapping to obtain the smallest, fastest software.
28-30
Range
Guard Bits
You can eliminate the possibility of overflow by appending the appropriate
number of guard bits to a binary word.
For a two’s complement signed value, the guard bits are filled with either 0’s
or 1’s depending on the value of the most significant bit (MSB). This is called
sign extension. For example, consider a 4-bit two’s complement number with
value 1011. If this number is extended in range to 7 bits with sign extension,
then the number becomes 1111101 and the value remains the same.
Guard bits are supported only for fractional data types. For both signed and
unsigned fractionals, the guard bits lie to the left of the default binary point.
max ( Q ) − min ( Q ) ,
where
⎧⎪ 0 unsigned,
min ( Q ) = ⎨ ws−1
⎩⎪−2 signed,
⎧⎪ 2ws − 1 unsigned,
max ( Q ) = ⎨
ws−1
⎩⎪ 2 −1 signed.
max ( V ) − min ( V ) ,
where
28-31
28 Arithmetic Operations
⎧⎪ B unsigned,
min ( V ) = ⎨
⎩⎪− F 2 ( 2
E ws−1 )
+B signed,
⎧⎪ F 2 ( 2 − 1) +B
E ws
unsigned,
max ( V ) = ⎨
⎪⎩ F 2 E ( 2ws−1 − 1) +B signed.
If the real-world value exceeds the limited range of the approximate value,
then the accuracy of the representation can become significantly worse.
28-32
Recommendations for Arithmetic and Scaling
• Your scaling choices depend on the particular design you are simulating.
28-33
28 Arithmetic Operations
Addition
Consider the addition of two real-world values:
Va = Vb + Vc .
These values are represented by the general [Slope Bias] encoding scheme
described in “Scaling” on page 27-6:
Vi = Fi 2 Ei Qi + Bi .
Fb Eb − Ea F B + Bc − Ba − Ea
Qa = 2 Qb + c 2 Ec − Ea Qc + b 2 .
Fa Fa Fa
28-34
Recommendations for Arithmetic and Scaling
Fc Ec − Ea
Qa = 2 Eb − Ea Qb + 2 Qc
Fa
or
Fb Eb − Ea
Qa = 2 Qb + 2 Ec − Ea Qc .
Fa
In most cases the input and output word sizes are much greater than one,
and the slope becomes
28-35
28 Arithmetic Operations
which depends only on the size of the input and output words. The
corresponding bias is
The value of the bias depends on whether the inputs and output are signed
or unsigned numbers.
If the inputs and output are all unsigned, then the minimum values for these
variables are all zero and the bias reduces to a particularly simple form:
Ba = Bb + Bc .
If the inputs and the output are all signed, then the bias becomes
( ) ( )
Ba ≈ Bb + Bc + Fb 2 Eb −2wsb −1 + 2wsb −1 + Fc 2 Ec −2wsc −1 + 2wsc −1 ,
Ba ≈ Bb + Bc .
Binary-Point-Only Scaling
For binary-point-only scaling, finding Qa results in this simple expression:
Qa = 2 Eb − Ea Qb + 2 Ec − Ea Qc .
This scaling choice results in only one addition and some bit shifting. The
avoidance of any multiplications is a big advantage of binary-point-only
scaling.
Note The subtraction of values produces results that are analogous to those
produced by the addition of values.
28-36
Recommendations for Arithmetic and Scaling
Accumulation
The accumulation of values is closely associated with addition:
Va _ new = Va _ old + Vb .
Fb Eb − Ea B
Qa _ new = Qa _ old + 2 Qb + b 2− Ea .
Fa Fa
Binary-Point-Only Scaling
For binary-point-only scaling, finding Qa_new results in this simple expression:
Qa _ new = Qa _ old + 2 Eb − Ea Qb .
This scaling option only involves one addition and some bit shifting.
Multiplication
Consider the multiplication of two real-world values:
Va = VbVc .
These values are represented by the general [Slope Bias] encoding scheme
described in “Scaling” on page 27-6:
Vi = Fi 2 Ei Qi + Bi .
28-37
28 Arithmetic Operations
Fb Fc Eb + Ec − Ea F B
Qa = 2 QbQc + b c 2 Eb − Ea Qb
Fa Fa
Fc Bb Ec − Ea B B − Ba − Ea
+ 2 Qc + b c 2 .
Fa Fa
Bc − Ec B
Qa = QbQc + 2 Qb + b 2− Eb Qc .
Fc Fb
max ( Va )
28-38
Recommendations for Arithmetic and Scaling
and
min ( Va ) .
where
Binary-Point-Only Scaling
For binary-point-only scaling, finding Qa results in this simple expression:
Qa = 2 Eb + Ec − Ea QbQc .
Gain
Consider the multiplication of a constant and a variable
Va = K Vb ,
where K is a constant called the gain. Since Va results from the multiplication
of a constant and a variable, finding Qa is a simplified version of the general
fixed-point multiplication formula:
⎛ KF 2 Eb ⎞ ⎛ KB − B ⎞
b b a
Qa = ⎜ ⎟Qb + ⎜ ⎟.
⎜ F 2 Ea ⎟ ⎜ F 2 Ea ⎟
⎝ a ⎠ ⎝ a ⎠
28-39
28 Arithmetic Operations
Note that the terms in the parentheses can be calculated offline. Therefore,
there is only one multiplication of a constant and a variable and one addition.
⎛ KF 2 Eb ⎞
⎜ b ⎟ = 2 EX Q X
⎜ F 2 Ea ⎟
⎝ a ⎠
⎛ KB − B ⎞
⎜ b a⎟
= 2 EY QY
⎜ F 2 Ea ⎟
⎝ a ⎠
Qa = 2 EX QX QB + 2 EY QY .
Qa = Qb .
28-40
Recommendations for Arithmetic and Scaling
Division
Division of values is an operation that should be avoided in fixed-point
embedded systems, but it can occur in places. Therefore, consider the division
of two real-world values:
Va = Vb Vc .
These values are represented by the general [Slope Bias] encoding scheme
described in “Scaling” on page 27-6:
Vi = Fi 2 Ei Qi + Bi .
Fb 2 Eb Qb + Bb Ba − Ea
Qa = − 2 .
Ec + Ea Ea
Fc Fa 2 Qc + Bc Fa 2 Fa
28-41
28 Arithmetic Operations
Qb Eb − Ec − Ea ( Bb Fb ) − Ec − Ea
Qa = 2 + 2 .
Qc Qc
max ( Va )
and
min ( Va ) .
28-42
Recommendations for Arithmetic and Scaling
Binary-Point-Only Scaling
For binary-point-only scaling, finding Qa results in this simple expression:
Qb Eb − Ec − Ea
Qa = 2 .
Qc
Note For the last two formulas involving Qa, a divide by zero and zero
divided by zero are possible. In these cases, the hardware will give some
default behavior but you must make sure that these default responses give
meaningful results for the embedded system.
Summary
From the previous analysis of fixed-point variables scaled within the general
[Slope Bias] encoding scheme, you can conclude
28-43
28 Arithmetic Operations
Introduction
To completely understand the results generated by fixed-point Simulink
blocks, you must be aware of these issues:
28-44
Parameter and Signal Conversions
The sections that follow describe parameter and signal conversions. “Rules
for Arithmetic Operations” on page 28-49 discusses arithmetic operations.
Parameter Conversions
Parameters of fixed-point blocks that accept numerical values are always
converted from double to a fixed-point data type. Parameters can be
converted to the input data type, the output data type, or to a data type
explicitly specified by the block. For example, the Discrete FIR Filter block
converts its Initial states parameter to the input data type, and converts its
Numerator coefficient parameter to a data type you explicitly specify via
the block dialog box.
28-45
28 Arithmetic Operations
Offline Conversions
An offline conversion is a conversion performed by your development platform
(for example, the processor on your PC), and not by the fixed-point processor
you are targeting. For example, suppose you are using a PC to develop a
program to run on a fixed-point processor, and you need the fixed-point
processor to compute
⎛ ab ⎞
y=⎜ ⎟ u = Cu
⎝ c ⎠
Signal Conversions
Consider the conversion of a real-world value from one fixed-point data type
to another. Ideally, the values before and after the conversion are equal.
Va = Vb ,
where Vb is the input value and Va is the output value. To see how the
conversion is implemented, the two ideal values are replaced by the general
[Slope Bias] encoding scheme described in “Scaling” on page 27-6:
Vi = Fi 2 Ei Qi + Bi .
Solving for the output data type’s stored integer value, Qa is obtained:
28-46
Parameter and Signal Conversions
Fb Eb − Ea B − Ba − Ea
Qa = 2 Qb + b 2
Fa Fa
= Fs 2 Eb − Ea Qb + Bnet ,
where Fs is the adjusted fractional slope and Bnet is the net bias. The offline
conversions and online conversions and operations are discussed below.
Offline Conversions
Both Fs and Bnet are computed offline using round-to-nearest and saturation.
Bnet is then stored using the output data type and Fs is stored using an
automatically selected data type.
Qa = Bnet .
2 The input integer value, Qb, is multiplied by the adjusted slope, Fs:
QRawProduct = Fs Qb .
3 The result of step 2 is converted to the modified output data type where the
slope is one and bias is zero:
Qa = QTemp + Qa .
28-47
28 Arithmetic Operations
Qa = convert ( Qb ) .
Exclusive use of binary-point-only scaling for both input signals and output
signals is a common way to eliminate mismatched slopes and biases, and
results in the most efficient simulations and generated code.
28-48
Rules for Arithmetic Operations
Introduction
Fixed-point arithmetic refers to how signed or unsigned binary words are
operated on. The simplicity of fixed-point arithmetic functions such as
addition and subtraction allows for cost-effective hardware implementations.
The sections that follow describe the rules that the Simulink software follows
when arithmetic operations are performed on inputs and parameters. These
rules are organized into four groups based on the operations involved:
addition and subtraction, multiplication, division, and shifts. For each
of these four groups, the rules for performing the specified operation are
presented with an example using the rules.
Computational Units
The core architecture of many processors contains several computational
units including arithmetic logic units (ALUs), multiply and accumulate units
(MACs), and shifters. These computational units process the binary data
directly and provide support for arithmetic computations of varying precision.
The ALU performs a standard set of arithmetic and logic operations as well as
division. The MAC performs multiply, multiply/add, and multiply/subtract
28-49
28 Arithmetic Operations
• If both numbers are positive and the result of their addition has a sign bit
of 1, then overflow has occurred; otherwise the result is correct.
• If both numbers are negative and the sign of the result is 0, then overflow
has occurred; otherwise the result is correct.
• If the numbers are of unlike sign, overflow cannot occur and the result is
always correct.
Va = ±Vb ± Vc ,
where Vb and Vc are the input values and Va is the output value. To see how
the summation is actually implemented, the three ideal values should be
replaced by the general [Slope Bias] encoding scheme described in “Scaling”
on page 27-6:
Vi = Fi 2 Ei Qi + Bi .
The equation in “Addition” on page 28-34 gives the solution of the resulting
equation for the stored integer, Qa. Using shorthand notation, that equation
becomes
28-50
Rules for Arithmetic Operations
where Fsb and Fsc are the adjusted fractional slopes and Bnet is the net bias.
The offline conversions and online conversions and operations are discussed
below.
Offline Conversions. Fsb, Fsc, and Bnet are computed offline using
round-to-nearest and saturation. Furthermore, Bnet is stored using the output
data type.
Qa = Bnet .
2 The first input integer value, Qb, is multiplied by the adjusted slope, Fsb:
QRawProduct = FsbQb .
3 The previous product is converted to the modified output data type where
the slope is one and the bias is zero:
Qa = Qa ± QTemp .
28-51
28 Arithmetic Operations
It is important to note that bit shifting, rounding, and overflow handling are
applied to the intermediate steps (3 and 4) and not to the overall sum.
The summation is shown in the following model for the input values 19.875,
5.4375, and 4.84375.
Applying the rules from the previous section, the sum follows these steps:
28-52
Rules for Arithmetic Operations
Qa = 00000.000.
2 The first number to be summed (19.875) has a fractional slope that matches
the output fractional slope. Furthermore, the binary points and storage
types are identical, so the conversion is trivial:
Qb = 10011.111,
QTemp = Qb .
Qa = Qa + QTemp = 10011.111.
Qc = 0101.0111,
QTemp = convert ( Qc )
QTemp = 00101.011.
Note that a loss in precision of one bit occurs, with the resulting value of
QTemp determined by the rounding mode. For this example, round-to-floor
is used. Overflow cannot occur in this case because the bits and binary
point are both shifted to the right.
Qa = Qa + QTemp
10011.111
+ 00101.011
=
11001.010 = 25.250.
28-53
28 Arithmetic Operations
Note that overflow did not occur, but it is possible for this operation.
Qd = 100.11011,
QTemp = convert ( Qd )
QTemp = 00100.110.
Note that a loss in precision of two bit occurs, with the resulting value of
QTemp determined by the rounding mode. For this example, round-to-floor
is used. Overflow cannot occur in this case because the bits and binary
point are both shifted to the right.
Qa = Qa + QTemp
11001.010
+ 00100.110
=
11110.000 = 30.000.
Note that overflow did not occur, but it is possible for this operation.
As shown here, the result of step 7 differs from the ideal sum:
10011.111
0101.0111
+ 100.11011
=
11110.001 = 30.125.
Blocks that perform addition and subtraction include the Sum, Gain, and
Discrete FIR Filter blocks.
28-54
Rules for Arithmetic Operations
Multiplication
The multiplication of an n-bit binary number with an m-bit binary number
results in a product that is up to m + n bits in length for both signed and
unsigned words. Most processors perform n-bit by n-bit multiplication and
produce a 2n-bit result (double bits) assuming there is no overflow condition.
Va = VbVc .
where Vb and Vc are the input values and Va is the output value. To see how
the multiplication is actually implemented, the three ideal values should be
replaced by the general [Slope Bias] encoding scheme described in “Scaling”
on page 27-6:
Vi = Fi 2 Ei Qi + Bi .
The solution of the resulting equation for the output stored integer, Qa, is
given below:
Fb Fc Eb + Ec − Ea F B
Qa = 2 QbQc + b c 2 Eb − Ea Qb
Fa Fa
Fc Bb Ec − Ea B B − Ba − Ea
+ 2 Qc + b c 2 .
Fa Fa
28-55
28 Arithmetic Operations
If both the inputs and the output have nonzero biases, then the operation is
broken down as follows:
V1Temp = V1 ,
V2Temp = V2 ,
V3Temp = V1TempV2Temp ,
V3 = V3Temp ,
where
E1 Temp
V1Temp = 2 Q1Temp ,
E2 Temp
V2Temp = 2 Q2Temp ,
E3 Temp
V3Temp = 2 Q3Temp .
VMax = F 2 E QMaxInteger + B,
28-56
Rules for Arithmetic Operations
VMin = F 2 E QMinInteger + B.
The data type and scaling of the temporary value must be able to represent
this range without overflow. Precision loss is possible, but overflow is never
allowed.
• Use a data type that leads to efficient operations.
This goal is relative to the target that you will use for production
deployment of your design. For example, suppose that you will implement
the design on a 16-bit fixed-point processor that provides a 32-bit long,
16-bit int, and 8-bit short or char. For such a target, preserving efficiency
means that no more than 32 bits are used, and the smaller sizes of 8 or 16
bits are used if they are sufficient to maintain precision.
• Maintain precision.
Ideally, every possible value defined by the original data type and scaling
is represented perfectly by the temporary variable. However, this can
require more bits than is efficient. Bits are discarded, resulting in a loss of
precision, to the extent required to preserve efficiency.
where QOriginal is an 8-bit, unsigned data type. For this data type,
QMaxInteger = 225,
QMinInteger = 0,
so
VMax = 211.75,
VMin = −43.25.
28-57
28 Arithmetic Operations
get full precision, the fixed exponent of the temporary variable has to be -2
or less. The Simulink software selects the least possible precision, which is
generally the most efficient, unless overflow issues arise. For a scaling of 2-2,
selecting signed 16-bit or signed 32-bit avoids overflow. For efficiency, the
Simulink software selects the smaller choice of 16 bits. If the original variable
is an input, then the equations to convert to the temporary variable are
uint8_T QOriginal ,
uint16_T QTemp ,
QTemp = (( uint16_T ) QOriginal )
2 − 173.
Fb Fc Eb + Ec − Ea
Qa = 2 QbQc .
Fa
Offline Conversions
The quantity
Fb Fc
FNet =
Fa
2 ENet QNet ,
where ENet and QNet are selected automatically to best represent FNet.
28-58
Rules for Arithmetic Operations
QRawProduct = QbQc .
To maintain the full precision of the product, the binary point of QRawProduct
is given by the sum of the binary points of Qb and Qc.
3 The multiplication
Q2 RawProduct = QTempQNet
is performed.
Qa = convert ( Q2 RawProduct ) .
Qa = 2 Eb + Ec − Ea QbQc .
Offline Conversions
28-59
28 Arithmetic Operations
QRawProduct = QbQc .
To maintain the full precision of the product, the binary point of QRawProduct
is given by the sum of the binary points of Qb and Qc.
Qa = convert ( QRawProduct ) .
Applying the rules from the previous section, the multiplication follows these
steps:
28-60
Rules for Arithmetic Operations
QRawProduct = 101.11
×10.011
101.11 ⋅ 2−3
101.11 ⋅ 2−2
+ 101.11 ⋅ 21
01101.10101 = 13.65625.
Note that the binary point of the product is given by the sum of the binary
points of the multiplied numbers.
3 The result of step 2 and the third number (1.8125) are multiplied:
QRawProduct = 01101.1010
×1.1101
1101.1010 ⋅ 2−4
1101.1010 ⋅ 2−2
1101.1010 ⋅ 2−1
+ 1101.1010 ⋅ 20
0011000.10110010 = 24.6953125.
Note that the binary point of the product is given by the sum of the binary
points of the multiplied numbers.
28-61
28 Arithmetic Operations
Qa = convert ( QRawProduct )
= 011000.1011 = 24.6875.
Blocks that perform multiplication include the Product, Discrete FIR Filter,
and Gain blocks.
Division
This section discusses the division of quantities with zero bias.
Note When any input to a division calculation has nonzero bias, the
operations performed exactly match those for multiplication described in
“Multiplication with Nonzero Biases and Mismatched Fractional Slopes” on
page 28-55.
Va = Vb Vc ,
where Vb and Vc are the input values and Va is the output value. To see
how the division is actually implemented, the three ideal values should be
replaced by the general [Slope Bias] encoding scheme described in “Scaling”
on page 27-6:
Vi = Fi 2 Ei Qi + Bi .
28-62
Rules for Arithmetic Operations
For the case where the slope adjustment factors are one and the biases are
zero for all signals, the solution of the resulting equation for the output stored
integer, Qa, is given by the following equation:
Qa = 2 Eb − Ec − Ea ( Qb Qc ) .
This equation involves an integer division and some bit shifts. If Ea > Eb–Ec,
then any bit shifts are to the right and the implementation is simple. However,
if Ea < Eb–Ec, then the bit shifts are to the left and the implementation can be
more complicated. The essential issue is that the output has more precision
than the integer division provides. To get full precision, a fractional division
is needed. The C programming language provides access to integer division
only for fixed-point data types. Depending on the size of the numerator,
you can obtain some of the fractional bits by performing a shift prior to the
integer division. In the worst case, it might be necessary to resort to repeated
subtractions in software.
28-63
28 Arithmetic Operations
( −4 )−( −4 )
Qa = 2−4 − ( Qb Qc )
=2 4
( Qb Qc ) .
Qa =
( 24 Qb ) ,
Qc
where the numerator uses the larger data type. If a larger data type was not
available, integer division combined with four repeated subtractions would
be used. Both approaches produce the same result, with the former being
more efficient.
Shifts
Nearly all microprocessors and digital signal processors support well-defined
bit-shift (or simply shift) operations for integers. For example, consider the
8-bit unsigned integer 00110101. The results of a 2-bit shift to the left and a
2-bit shift to the right are shown in the following table.
28-64
Rules for Arithmetic Operations
You can perform a shift using the Simulink Shift Arithmetic block. Use this
block to perform a bit shift, a binary point shift, or both
The Shift Arithmetic block performs an arithmetic shift right and, therefore,
recycles the most significant bit for each bit shift right. For example, given
the fixed-point number 11001.011 (-6.625), a bit shift two places to the right
with the binary point unmoved yields the number 11110.010 (-1.75), as shown
in the model below:
To perform a logical shift right on a signed number using the Shift Arithmetic
block, use the Data Type Conversion block to cast the number as an unsigned
number of equivalent length and scaling, as shown in the following model.
The model shows that the fixed-point signed number 11001.001 (-6.625)
becomes 00110.010 (6.25).
28-65
28 Arithmetic Operations
28-66
Conversions and Arithmetic Operations
Note If a block can perform all four arithmetic operations, then the rules for
multiplication and division are applied first. The Discrete FIR Filter block
is an example of this.
Suppose you configure the Discrete FIR Filter block for two outputs, where
the first output is given by
y1 ( k ) = 13 ⋅ u ( k ) + 11 ⋅ u ( k − 1) − 7 ⋅ u ( k − 2 ) ,
y2 ( k ) = 6 ⋅ u ( k ) − 5 ⋅ u ( k − 1) .
Additionally, the initial values of u(k–1) and u(k–2) are given by 0.8 and 1.1,
respectively, and all inputs, parameters, and outputs have binary-point-only
scaling.
To configure the Discrete FIR Filter block for this situation, on the Main pane
of its dialog box, you must specify the Coefficients parameter as [13 11 -7;
6 -5 0] and the Initial states parameter as [0.8 1.1], as shown here.
28-67
28 Arithmetic Operations
Similarly, configure the options on the Data Types pane of the block dialog
box to appear as follows:
28-68
Conversions and Arithmetic Operations
The Discrete FIR Filter block performs parameter conversions and block
operations in the following order:
The Initial states parameter is converted offline from doubles to the input
data type using round-to-nearest and saturation.
2 The coefficients and inputs are multiplied together for the initial time step
for both outputs. For y1(0), the operations 13·u(0), 11·0.8, and –7·1.1 are
performed, while for y2(0), the operations 6·u(0) and –5·0.8 are performed.
28-69
28 Arithmetic Operations
28-70
29
Realization Structures
Introduction
This chapter investigates how you can realize fixed-point digital filters using
Simulink blocks and the Fixed-Point Designer software.
The Fixed-Point Designer software addresses the needs of the control system,
signal processing, and other fields where algorithms are implemented on
fixed-point hardware. In signal processing, a digital filter is a computational
algorithm that converts a sequence of input numbers to a sequence of
output numbers. The algorithm is designed such that the output signal
meets frequency-domain or time-domain constraints (desirable frequency
components are passed, undesirable components are rejected).
29-2
Realizing Fixed-Point Digital Filters
In the world of fixed-point numbers, where precision and range are limited,
the differences in the realization results can be very large. Therefore, you
must carefully select the data type, word size, and scaling for each realization
element such that results are accurately represented. To assist you with this
selection, design rules for modeling dynamic systems with fixed-point math
are provided in “Targeting an Embedded Processor” on page 29-4.
29-3
29 Realization Structures
Introduction
The sections that follow describe issues that often arise when targeting a
fixed-point design for use on an embedded processor, such as some general
assumptions about integer sizes and operations available on embedded
processors. These assumptions lead to design issues and design rules that
might be useful for your specific fixed-point design.
Size Assumptions
Embedded processors are typically characterized by a particular bit size. For
example, the terms “8-bit micro,” “32-bit micro,” or “16-bit DSP” are common.
It is generally safe to assume that the processor is predominantly geared to
processing integers of the specified bit size. Integers of the specified bit size
are referred to as the base data type. Additionally, the processor typically
provides some support for integers that are twice as wide as the base data
type. Integers consisting of double bits are referred to as the accumulator
data type. For example a 16-bit micro has a 16-bit base data type and a 32-bit
accumulator data type.
Although other data types may be supported by the embedded processor, this
section describes only the base and accumulator data types.
Operation Assumptions
The embedded processor operations discussed in this section are limited to the
needs of a basic simulation diagram. Basic simulations use multiplication,
addition, subtraction, and delays. Fixed-point models also need shifts to
do scaling conversions. For all these operations, the embedded processor
29-4
Targeting an Embedded Processor
should have native instructions that allow the base data type as inputs.
For accumulator-type inputs, the processor typically supports addition,
subtraction, and delay (storage/retrieval from memory), but not multiplication.
Design Rules
The important design rules that you should be aware of when modeling
dynamic systems with fixed-point math follow.
You can insert blocks to convert inputs from the accumulator type to the base
type prior to Product or Gain blocks, if necessary.
• The Unit Delay essentially stores a variable’s value to RAM and, one time
step later, retrieves that value from RAM. Because the value must be in
memory from one time step to the next, the RAM must be exclusively
dedicated to the variable and can’t be shared or used for another purpose.
29-5
29 Realization Structures
Alternatively, the products can be summed using the accumulator data type,
and the final sum can be converted to the base data type. Round-off error is
introduced in just one point and the precision is generally better. The cost of
doing an addition or subtraction using accumulator-type numbers is slightly
more expensive, but if there is justification, it is usually worth the cost.
29-6
Canonical Forms
Canonical Forms
In this section...
“Canonical Forms” on page 29-7
“Direct Form II” on page 29-8
“Series Cascade Form” on page 29-13
“Parallel Form” on page 29-15
Canonical Forms
The Fixed-Point Designer software does not attempt to standardize on
one particular fixed-point digital filter design method. For example, you
can produce a design in continuous time and then obtain an “equivalent”
discrete-time digital filter using one of many transformation methods.
Alternatively, you can design digital filters directly in discrete time. After you
obtain a digital filter, it can be realized for fixed-point hardware using any
number of canonical forms. Typical canonical forms are the direct form, series
form, and parallel form, each of which is outlined in the sections that follow.
For a given digital filter, the canonical forms describe a set of fundamental
operations for the processor. Because there are an infinite number of ways
to realize a given digital filter, you must make the best realization on a
per-system basis. The canonical forms presented in this chapter optimize the
implementation with respect to some factor, such as minimum number of
delay elements.
In general, when choosing a realization method, you must take these factors
into consideration:
• Cost
The cost of the realization might rely on minimal code and data size.
• Timing constraints
Real-time systems must complete their compute cycle within a fixed
amount of time. Some realizations might yield faster execution speed on
different processors.
29-7
29 Realization Structures
The Fixed-Point Designer software allows you to evaluate various digital filter
realization methods in a simulation environment. Following the development
cycle outlined in “Developing and Testing Fixed-Point Systems” on page
26-17, you can fine-tune the realizations with the goal of reducing the cost
(code and data size) or increasing signal quality. After you have achieved the
desired performance, you can use the Simulink Coder product to generate
rapid prototyping C code and evaluate its performance with respect to your
system’s real-time timing constraints. You can then modify the model based
upon feedback from the rapid prototyping system.
The presentation of the various realization structures takes into account that
a summing junction is a fundamental operator, thus you may find that the
structures presented here look different from those in the fixed-point filter
design literature. For each realization form, an example is provided using the
transfer function shown here:
Direct Form II
In general, a direct form realization refers to a structure where the coefficients
of the transfer function appear directly as Gain blocks. The direct form
II realization method is presented as using the minimal number of delay
elements, which is equal to n, the order of the transfer function denominator.
29-8
Canonical Forms
You can derive the canonical direct form II realization by writing the
discrete-time transfer function with input e(z) and output u(z) as
u ( z) u ( z) h ( z)
= ⋅
e ( z) h ( z) e ( z)
(
= b0 + b1 z−1 + + bm z− m) 1
1 + a1 z + a2 z−2 + an z− n
−1
.
u( z ) h( z )
h( z ) e( z )
29-9
29 Realization Structures
Combining these two block diagrams yields the direct form II diagram shown
in the following figure. Notice that the feedforward part (top of block diagram)
contains the numerator coefficients and the feedback part (bottom of block
diagram) contains the denominator coefficients.
29-10
Canonical Forms
fxpdemo_direct_form2
29-11
29 Realization Structures
29-12
Canonical Forms
u ( z)
Hi ( z ) = = H1 ( z ) ⋅ H2 ( z ) ⋅ H3 ( z ) H p ( z ) .
e ( z)
Factoring H(z) into Hi(z) where i = 1,2,3,...,p can be done in a number of ways.
Using the poles and zeros of H(z), you can obtain Hi(z) by grouping pairs
of conjugate complex poles and pairs of conjugate complex zeros to produce
second-order transfer functions, or by grouping real poles and real zeros to
produce either first-order or second-order transfer functions. You could also
group two real zeros with a pair of conjugate complex poles or vice versa.
Since there are many ways to obtain Hi(z), you should compare the various
groupings to see which produces the best results for the transfer function
under consideration.
H ( z ) = H1 ( z ) H2 ( z ) H p ( z )
j
1 + bi z−1 p
1 + ei z−1 + fi z−2
=∏ ∏ .
i=1 1 + ai z−1 i= j +1 1 + ci z−1 + di z−2
You must also take into consideration that the ordering of the individual
Hi(z)’s will lead to systems with different numerical characteristics. You
might want to try various orderings for a given set of Hi(z)’s to determine
which gives the best numerical characteristics.
29-13
29 Realization Structures
29-14
Canonical Forms
fxpdemo_series_cascade_form
Parallel Form
In the canonical parallel form, the transfer function H(z) is expanded into
partial fractions. H(z) is then realized as a sum of a constant, first-order, and
second-order transfer functions, as shown:
29-15
29 Realization Structures
u ( z)
Hi ( z ) = = K + H1 ( z ) + H2 ( z ) + + H p ( z ) .
e ( z)
This expansion, where K is a constant and the Hi(z) are the first- and
second-order transfer functions, follows.
j p
H ( z ) = K + ∑ Hi ( z ) + ∑ Hi ( z )
i=1 i= j +1
j
bi p
ei + fi z−1
= K +∑ + ∑ .
i=1 1 + ai z−1 i= j +1 1 + ci z−1 + di z−2
29-16
Canonical Forms
29-17
29 Realization Structures
fxpdemo_parallel_form
29-18
Canonical Forms
29-19
29 Realization Structures
29-20
30
Fixed-Point Advisor
Introduction
Using the Fixed-Point Advisor, you can prepare a model for conversion from a
floating-point model or subsystem to an equivalent fixed-point representation.
After preparing the model for conversion, use the Fixed-Point Tool to obtain
initial fixed-point data types and then refine these data types.
Best Practices
30-2
Preparation for Fixed-Point Conversion
This practice provides you with a back up in case of error and a baseline for
testing and validation.
Convert Subsystems
Convert subsystems within your model, rather than the entire model. This
practice saves time and unnecessary conversions.
The ultimate goal is to replace all blocks that do not support fixed-point data
types. You must eventually replace blocks that you isolate with Data Type
Conversion blocks with blocks that do support fixed-point data types.
30-3
30 Fixed-Point Advisor
• You already know the fixed-point data types that you want to use for a
particular block.
For example, the block is modeling a real-world component. Set up the
block to allow for known hardware limitations, such as restricting outputs
to integer values.
Specify the output data type of the block explicitly and select Lock output
data type setting against changes by the fixed-point tools.
• You are debugging a model and know that a particular block accepts only
certain data types.
Specify the output data type of upstream blocks explicitly and select Lock
output data type setting against changes by the fixed-point tools.
30-4
Preparation for Fixed-Point Conversion
3 In the Fixed-Point Advisor window, on the left pane, select the Fixed-Point
Advisor folder.
The Fixed-Point Advisor runs the tasks in order until a task fails. A
waitbar is displayed while each task runs.
5 Review the results. If a task fails because input parameters are not
specified, select an Input Parameter. Then continue running to failure
by right-clicking the task and selecting Continue from the context menu.
If the task fails for a different reason, fix the task as described in “Fix a
Task Failure” on page 30-5.
30-5
30 Fixed-Point Advisor
• Follow the instructions in the Analysis Result box. Use this method to fix
failures individually. See “Manually Fixing Failures” on page 30-6.
• Use the Action box. Use this method to automatically fix all failures. See
“Automatically Fixing Failures” on page 30-6.
• Use the Model Advisor Results Explorer. Use this method to batch fix
failures. See “Batch Fixing Failures” on page 30-7
Note A warning result is meant for your information. You can choose to fix
the reported issue or move on to the next task.
1 Optionally, save a restore point so you can undo the changes that you
make. For more information, see “Save a Restore Point” on page 30-8.
2 In the Analysis Result box, review the recommended actions. Use the
information to make changes to your model.
3 To verify that the task now passes, in the Analysis box, click Run This
Task.
Automatically fix all failures within a task using the following steps:
30-6
Preparation for Fixed-Point Conversion
1 Optionally, save a restore point so you can undo the changes that you
make. For more information, see “Save a Restore Point” on page 30-8.
3 To verify that the task now passes, in the Analysis box, click Run This
Task.
1 Optionally, save a restore point so you can undo the changes that you
make. For more information, see “Save a Restore Point” on page 30-8.
4 When you finish making changes, in the Fixed-Point Advisor window, click
Run This Task to see if the changes you made result in the task passing.
Continue fixing failures and rerunning the task until the task passes.
Restore Points
The Fixed-Point Advisor provides a model and data restore point capability for
reverting changes that you made in response to advice from the Fixed-Point
Advisor. A restore point is a snapshot in time of the model, base workspace,
and Fixed-Point Advisor.
Caution A restore point saves only the current working model, base
workspace variables, and Fixed-Point Advisor tree. It does not save other
items, such as libraries and referenced submodels.
To learn how to save a restore point, see “Save a Restore Point” on page 30-8.
To learn how to load a restore point, see “Load a Restore Point” on page 30-9.
30-7
30 Fixed-Point Advisor
• Before applying changes to your model that might cause update diagram
failure. For example, before applying proposed data types in task 3.1.
• Before attempting to fix failures.
1 From the main menu, select File > Save Restore Point As.
30-8
Preparation for Fixed-Point Conversion
2 In the Save Model and Data Restore Point dialog box, in the Name
field, enter a name for the restore point.
3 In the Description field, you can optionally add a description to help you
identify the restore point.
4 Click Save.
The Fixed Point Advisor saves a restore point of the current model, base
workspace, and Fixed Point Advisor status.
Note To quickly save a restore point, go to File > Save Restore Point.
The Fixed Advisor saves a restore point with the name autosaven. n is the
sequential number of the restore point. If you use this method, you cannot
change the name of, or add a description to, the restore point.
• A task fails and you cannot continue the conversion. In this case, load a
restore point saved earlier in the run to avoid rerunning all the previous
tasks.
• You want to revert changes you made in response to advice from the
Fixed-Point Advisor.
30-9
30 Fixed-Point Advisor
2 In the Load Model and Data Restore Point dialog box, select the restore
point that you want.
3 Click Load.
The Model Advisor issues a warning that the restoration will overwrite
the current model and workspace.
The Fixed Point Advisor reverts the model, base workspace, and Fixed
Point Advisor status.
30-10
Converting a Model from Floating- to Fixed-Point Using Simulation Data
30-11
30 Fixed-Point Advisor
1 For the purpose of this tutorial, run the tasks in the Fixed-Point Advisor
Prepare Model for Conversion folder one at a time. In the left pane,
30-12
Converting a Model from Floating- to Fixed-Point Using Simulation Data
select Verify model simulation settings and, in the right pane, click
Run This Task.
This task validates that model simulation settings allow signal logging and
disable data type override to facilitate conversion to fixed point. These
settings ensure that fixed-point data can be logged in downstream tasks.
This task identifies blocks that do not support fixed-point data types.
The Fixed-Point Advisor cannot convert these blocks. To complete the
conversion of your model, replace these blocks with Simulink built-in
blocks that do support fixed-point data types. If a replacement block is not
available, you can temporarily isolate the unsupported block with Data
Type Conversion blocks.
The task fails because the model contains a block that does not support
fixed-point data types.
4 Fix the failure by replacing the TrigFcn block with the provided
replacement:
a Click the Preview link to view the replacement block.
b Click the link to the original block and view its settings.
c Double-click the replacement block and verify its settings match the
settings of the original block.
Note If the settings on the replacement block differ from the settings
on the original block, set up the replacement block to match the original
block.
30-13
30 Fixed-Point Advisor
5 Select and run Set up signal logging. Because you are using simulation
minimum and maximum data, you must specify at least one signal to use
in analysis and comparison in downstream checks. At a minimum, you
should log the unique input and output signals.
The task runs and the Fixed-Point Advisor warns that signal logging is
not specified for any signals.
6 Because you want to propose data types based on simulation data, fix the
warning:
a Click the Explore Result button.
b In the Model Advisor Result Explorer, select the signals that you want to
log and select the EnableLogging check box.
For this tutorial, log the signals connected to the Inport and Outport
blocks:
• Ctr_in
• Ctr_out
c Close the Model Advisor Result Explorer.
The task passes because signal logging is enabled for at least one signal.
The task runs and the Fixed-Point Advisor warns that logging is not
enabled.
30-14
Converting a Model from Floating- to Fixed-Point Using Simulation Data
The Modify All action configures the model to the settings recommended
in the Analysis Result. The Action Result box displays a table of changes.
The task passes and the tool stores the results in a run named
FPA_Reference. You can view these results in the Fixed-Point Tool
Contents pane.
11 Select and run Implement logic signals as Boolean data. This task
verifies that Model Configuration Parameters > Optimization >
Implement logic signals as Boolean data is selected. If it is cleared, the
code generated in downstream checks is not optimized.
30-15
30 Fixed-Point Advisor
Note This is a Simulink check. For more information, see “Check bus
usage” in the Simulink documentation.
13 Select and run Simulation range checking. This tasks verifies that
the Model Configuration Parameters > Diagnostics > Simulation
range checking option is not set to none. A warning is displayed because
Simulation range checking is currently set to none. The recommended
setting is warning so that warnings are generated when signals exceed the
specified minimum or maximum values.
14 Fix the warning by applying the recommended setting using the Modify
All button. Rerun the task.
15 Select and run Check for implicit signal resolution. This task checks
for models that use implicit signal resolution. To use the Fixed-Point
Advisor for Simulink signal object scaling, turn off implicit signal resolution
by setting the Diagnostics > Data Validity > Signal resolution
property in the Configuration Parameters dialog box to Explicit only.
Enforce resolution for each of the signals and states that currently resolve
successfully. For more information, see “Signal resolution” in the Simulink
documentation.
The task passes because the model contains no Simulink signal objects.
The run to failure action has completed for the Prepare Model for
Conversion folder. At this point, you can review the results report found at
the folder level, or continue to the next folder.
30-16
Converting a Model from Floating- to Fixed-Point Using Simulation Data
1 Right-click Prepare for Data Typing and Scaling and select Run to
Failure.
The Fixed-Point Advisor runs the Review locked data type settings
task. This task identifies blocks that have their data type settings locked
down, which excludes them from automatic data typing.
The task passes because it finds no blocks with locked data types.
2 The Fixed-Point Advisor runs the Remove output data type inheritance
task. This task identifies blocks with the Output data type property set to
Inherit. Inherited data types might lead to data type propagation errors.
The task fails because some blocks in the model have inherited output
data types.
3 Fix the failure using the Modify All button to explicitly configure the
output data types to the recommended values. Rerun the task.
4 Continue running to failure. Relax input data type settings runs. This
task identifies blocks with input data type constraints that might lead to
data type propagation errors.
The task passes because all blocks have flexible input data types.
5 Verify Stateflow charts that have strong data typing with Simulink
runs. This task verifies that all Stateflow charts are configured to have
strong data typing with Simulink I/O.
The task passes because the model does not have any Stateflow charts.
30-17
30 Fixed-Point Advisor
The task passes because the model contains no resolved Simulink signal
objects.
The task fails because you must specify a default data type for floating-point
data types that is suitable for the chosen hardware.
10 Fix the failure by setting Default data type for all floating-point
signals to int16.
The software uses this default data type for all output signals. The
Fixed-Point Advisor proposes the Same as embedded hardware integer
setting, which is int32. However, use int16 because the model performs
many multiplications and you want the product to fit into int32.
30-18
Converting a Model from Floating- to Fixed-Point Using Simulation Data
The Fixed-Point Advisor warns you that you have not specified any
minimum and maximum values. Optimally, specify block output and
parameter minimum and maximum values for, at minimum, the Inport
blocks in the system. You can specify the minimum and maximum values
for any block in this step. Typically, these values are determined during
the design process based on the system that you are creating.
13 Fix the warning by specifying minimum and maximum values for Inport
blocks:
a Click the Explore Result button.
The Model Advisor Result Explorer opens, displaying the Inport blocks
that do not have an output minimum and maximum specified.
b On the Model Advisor Result Explorer center pane, select Ctr_in. For
the purpose of this tutorial, specify the output minimum and maximum
values for this block. Set OutMin to -5 and set OutMax to 5.
c Close the Model Advisor Result Explorer.
The task passes because minimum and maximum values are specified
for all Inport blocks.
e For the purpose of this tutorial, do not specify other minimum and
maximum values for other blocks.
f Review the results report found at the folder level.
15 On the Fixed-Point Tool Contents pane, examine the results for the
simulation reference run. One of the TrigFcn block outputs overflowed
multiple times, indicating that the fixed-point settings on this block are
not suitable for the input range. To refine the fixed-point data types,
first run the model with a global override of the fixed-point data types
using double-precision numbers to avoid quantization effects. This action
provides a floating-point benchmark that represents the ideal output.
Then, propose new data types based on these “ideal” results.
30-19
30 Fixed-Point Advisor
The Fixed-Point Tool proposes new data types for objects in the model
and updates the results on the Contents pane.
2 In the Fixed-Point Tool, set the Column View to Automatic Data Typing
with Simulation Min/Max View to display information relevant to the
proposal. The tool displays the proposed scaling in the ProposedDT
column in the Contents pane.
3 Examine the results to resolve any conflicts and to ensure that you want
to accept the proposed data type for each result.
In the Fixed-Point Tool toolbar, select Show > Conflicts with proposed
data types.
The Fixed-Point Tool detects no conflicts, so you are ready to apply the new
data types as described in “Apply the New Fixed-Point Data Types” on
page 32-25.
30-20
Converting a Model from Floating- to Fixed-Point Using Simulation Data
The tool has set all the specified data types to the proposed types.
The Simulink software simulates the model using the new fixed-point
settings that you applied in the previous step and stores the results in
the NoOverride run.
4 Examine the results. Because the tool did not propose new data types for
the TrigFcn block, this block still overflows.
30-21
30 Fixed-Point Advisor
30-22
31
Fixed-Point Tool
Fixed-Point Tool
In this section...
“Introduction to the Fixed-Point Tool” on page 31-2
“Using the Fixed-Point Tool” on page 31-2
31-2
Fixed-Point Tool
If you want to open the tool programmatically, use the fxptdlg function.
For more information, see fxptdlg.
31-3
31 Fixed-Point Tool
31-4
Run Management
Run Management
In this section...
“Run Management” on page 31-5
“Why Use Shortcuts to Manage Runs” on page 31-7
“When to Use Shortcuts to Manage Runs” on page 31-7
“Add Shortcuts” on page 31-8
“Edit Shortcuts” on page 31-9
“Delete Shortcuts” on page 31-10
“Capture Model Settings Using the Shortcut Editor” on page 31-10
Run Management
The Fixed-Point Tool supports multiple runs. Each run uses one set of model
settings to simulate the model or to derive or propose data types. You can:
You can easily switch between different run setups using shortcuts.
Alternatively, you can manually manage runs.
31-5
31 Fixed-Point Tool
• The ability to add and edit custom shortcuts. The shortcuts are saved with
the model so that you define them once and then reuse them multiple times.
Use the Shortcut Editor to create or edit shortcuts and to add and organize
shortcut buttons in the Fixed-Point Tool Shortcuts to set up runs pane.
31-6
Run Management
Shortcuts:
31-7
31 Fixed-Point Tool
To ... Use...
Add Shortcuts
1 On the Fixed-Point Tool Shortcuts to set up runs pane, click Add/Edit
shortcuts.
2 For each subsystem that you want to specify a shortcut for, on the Shortcut
Editor Model Hierarchy pane, select the subsystem:
a In the Name of shortcut field, enter the shortcut name.
31-8
Run Management
Edit Shortcuts
1 On the Fixed-Point Tool Shortcuts to set up runs pane, click Add/Edit
shortcuts.
2 In the Shortcut Editor, from the Name of shortcut list, select the shortcut
that you want to edit.
4 If you do not want this shortcut to modify the existing data type override
settings on the model, clear Allow modification of data type override
settings.
5 If you do not want this shortcut to modify the run name on the model, clear
Allow modification of run name.
e Click Apply.
31-9
31 Fixed-Point Tool
8 Optionally, if you want the Fixed-Point Tool to display a button for this new
shortcut, use the right arrow to move the shortcut to the list of shortcuts to
display. Use the up and down arrows to change the order of the shortcut
buttons.
Delete Shortcuts
To delete a shortcut from a model:
31-10
Run Management
5 Click Apply.
31-11
31 Fixed-Point Tool
31-12
Debug a Fixed-Point Model
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_fixedpoint_debug
3 In the Fixed-Point Tool, set up a shortcut for the initial system settings:
31-13
31 Fixed-Point Tool
The editor sets the Run name for this shortcut to Setting A.
c Set Fixed-point instrumentation mode to Minimums, maximums and
overflows.
d Set Data type override to Use local settings.
e Click Apply.
5 Use this shortcut to set up a run. Use the settings to simulate the model.
31-14
Debug a Fixed-Point Model
quantization effects for this subsystem. If overflows occur, you can deduce
that there are issues with the fixed-point settings in subsystem Math1.
Next, simulate the model using local settings on Math2 and doubles override
on Math1. If overflows occur for this simulation, there are problems with the
fixed-point settings for subsystem Math2.
Setting Up Shortcuts
2 On the Manage shortcuts pane, add Setting B and Setting C to the list
of buttons to display in the Fixed-Point Tool.
31-15
31 Fixed-Point Tool
31-16
Debug a Fixed-Point Model
Tip In the Fixed-Point Tool Contents pane, click Run to sort the
results in this column.
No overflows occur, which indicates that the settings on Math1 are not
causing the overflows.
31-17
31 Fixed-Point Tool
2 On the Automatic data typing for selected system pane, click the
Propose fraction lengths button.
3 In the Propose Data Types dialog box, select Setting B as the run to use
for proposing data types and click OK. This run simulated Math2 with
double override to obtain the ’ideal’ behavior of the subsystem based on the
simulation results for both input stimuli.
The Fixed-Point Tool proposes new fixed-point data types for the objects in
subsystem Math2 to avoid numerical issues such as overflows.
5 Because the Fixed-Point Tool marked all the proposed results with a green
icon to indicate that the proposed data types pose no issues for these
objects, accept the proposals.
In the Automatic data typing for selected system pane, click the
Apply accepted fraction lengths button.
31-18
Debug a Fixed-Point Model
The Simulink software simulates the model using the new fixed-point
settings. Afterward, the Fixed-Point Tool displays the simulation results
in run Setting A2. No overflows or saturations occur indicating that the
model can now handle the full input range.
31-19
31 Fixed-Point Tool
31-20
Logging Simulation Ranges for Referenced Models
The tool logs and displays the results for each instance of the referenced
model. For example, here are the results for the first instance of the
referenced model ex_sum1 in ex_multi_instance.
31-21
31 Fixed-Point Tool
In the referenced model node, the tool displays the union of the results for
each instance of the referenced model.
31-22
Logging Simulation Ranges for Referenced Models
When you change the fixed-point instrumentation and data type override
settings for any instance of a referenced model, the settings change on all
instances of the model and on the referenced model itself.
See Also
• “Log Simulation Ranges for Referenced Models” on page 31-24
31-23
31 Fixed-Point Tool
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_mdlref_controller
31-24
Log Simulation Ranges for Referenced Models
The Fixed-Point Tool opens. In its Model Hierarchy pane, the tool
displays two model nodes, one for the ex_mdlref_controller model
31-25
31 Fixed-Point Tool
showing that this model contains a Model block that refers to the
ex_controller model, and another for the ex_controller model itself.
5 In the Fixed-Point Tool Data collection pane, set Store results in run to
initial_run and then click Apply.
31-26
Log Simulation Ranges for Referenced Models
Providing a unique name for the run avoids accidentally overwriting results
from previous runs and enables you to identify the run more easily.
6 The fixed-point instrumentation and data type override settings for the
top-level model do not affect the settings in the referenced model. In the
Fixed-Point Tool Model Hierarchy pane, select the ex_controller model
and verify that:
• Fixed-point instrumentation mode is set to Minimums, maximums
and overflows
• Data type override is set to Use local settings
The Simulation Data Inspector tool opens. You can use this tool to inspect
and compare signals in your model.
The Fixed-Point tool displays the results for the referenced model and
highlights the Up Cast block in red to indicate that there is an issue with
this result. The Saturations column for this result shows that the block
saturated 23 times, which indicates poor scaling.
31-27
31 Fixed-Point Tool
Next, use data type override mode to perform a global override of the
fixed-point data types and scaling using double-precision numbers to
avoid quantization effects. Later, you use these simulation results when
performing automatic data typing.
1 In the Settings for selected system pane, set Data type override to
Double.
Setting data type override for the top-level model avoids data type
propagation issues when you simulate the model.
31-28
Log Simulation Ranges for Referenced Models
Now you are ready to propose data types based on the simulation results
from the doubles override run.
31-29
31 Fixed-Point Tool
2 In the Automatic data typing for selected system pane, click the
Configure link and verify that Propose fraction lengths for specified
word lengths is selected.
3 In the same pane, specify the Safety margin for simulation min/max
(%) parameter as 20 and click Apply.
5 In the Propose Data Types dialog box, select double_run and click OK.
The Fixed-Point Tool analyzes the scaling of all fixed-point blocks whose:
• Lock output data type setting against changes by the fixed-point
tools parameter is not selected.
31-30
Propose Data Types for a Referenced Model
The Fixed-Point Tool uses the minimum and maximum values stored in
the selected run to propose each block’s scaling such that the precision is
maximized while the full range of simulation values is spanned. The tool
displays the proposed scaling in its Contents pane. Now, it displays the
Automatic Data Typing View to provide information, such as ProposedDT,
ProposedMin, ProposedMax, which are relevant at this stage of the
fixed-point conversion.
6 Review the scaling that the Fixed-Point Tool proposes. You can choose to
accept the scaling proposal for each block by selecting the corresponding
Accept check box in the Contents pane. By default, the Fixed-Point Tool
accepts all scaling proposals that differ from the current scaling. For this
example, verify that the Accept check box associated with the active run is
selected for each of the Controller subsystem’s blocks.
The Fixed-Point Tool does not propose a data type for Combine
Terms:Accumulator and displays n/a in the ProposedDT column. The
tool does not propose a data type because the SpecifiedDT is Inherit:
Inherit via internal rule. To view more information about a proposal,
click the Show details for selected result button .
31-31
31 Fixed-Point Tool
button .
The Fixed-Point Tool applies to the scaling proposals that you accepted
in the previous step.
9 In the Model Hierarchy pane, select the ex_controller model and set its
Data type override parameter as Use local settings and click Apply.
11 Use the Simulation Data Inspector to plot the Analog Plant output for the
floating-point and fixed-point runs and the difference between them.
The difference plot shows that the difference between the floating-point
signal and the fixed-point signal is within the specified tolerance of 0.04.
31-32
Propose Data Types for a Referenced Model
31-33
31 Fixed-Point Tool
Note The software does not log simulation minimum and maximum values
for MATLAB Function blocks used as a reference (library) block or in a
referenced model.
See Also
• “Log Simulation Ranges for MATLAB Function Block” on page 31-35
31-34
Log Simulation Ranges for MATLAB Function Block
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_matlab_function_block_logging
31-35
31 Fixed-Point Tool
31-36
Log Simulation Ranges for MATLAB Function Block
The Simulink software simulates the model using the specified fixed-point
instrumentation and local data type settings.
The Variables tab displays the simulation minimum and maximum values
for the MATLAB Function block input, output, and variables.
31-37
31 Fixed-Point Tool
31-38
View Signal Names in Fixed-Point Tool
3 Click .
4 Optionally, use the up and down arrow buttons to change the display order
for the columns.
31-39
31 Fixed-Point Tool
6 If a signal has a name, the Fixed-Point Tool displays the name in the
Contents pane.
31-40
32
Convert Floating-Point
Model to Fixed Point
Learning Objectives
In this example, you learn how to:
32-2
Learning Objectives
For floating-point inheritance blocks when inputs are floating point, all
internal and output data types are floating point. The model in this
example uses a Discrete Filter block, which is a floating-point inheritance
block.
32-3
32 Convert Floating-Point Model to Fixed Point
Model Description
In this section...
“Model Overview” on page 32-4
“Model Set Up” on page 32-5
Model Overview
This example uses the ex_fixed_point_workflow model.
This configuration allows you to modify the inputs and collect simulation data
for multiple stimuli. You can then examine the behavior of the subsystem
32-4
Model Description
with different input ranges and scale your fixed-point data types to provide
maximum precision while accommodating the full simulation range.
Model Set Up
The model consists of the following blocks and subsystem.
Source
Controller Subsystem
The Controller Subsystem consists of:
• Discrete Filter
The Discrete Filter block filters the Repeating table Source signal. The
Discrete Filter is a floating-point inheritance block. For floating-point
inheritance blocks, when inputs are floating-point, all internal and output
data types are floating point.
• Chart
32-5
32 Convert Floating-Point Model to Fixed Point
The Chart consists of aStateflow Chart block which converts the Sine Wave
input to a positive output and multiplies it by 3.
• Lookup Table for Chart
The Lookup Table for Chart block is the first of two identical n-D Lookup
Table blocks. This block receives the output from the Chart and, at each
breakpoint, outputs the input multiplied by 10.
• Gain
The Gain block multiplies the Sine Wave input by -3.
• Lookup Table for Gain
The Lookup Table for Gain block is a n-D Lookup Table block. It receives
the output from the Gain block and, at each breakpoint, outputs its input
multiplied by 10.
• Sum for Chart
This Sum block adds the outputs from the Discrete Filter and Lookup Table
for Chart blocks and outputs the result to the Scope block.
• Sum for Gain
This Sum block adds the outputs from the Discrete Filter and Lookup Table
for Gain blocks and outputs the result to the Scope block.
Scope
• Scope
The model includes a Scope block that displays the Controller Subsystem
output signals.
32-6
Before You Begin
Step through the Fixed-Point Advisor tasks that prepare the floating-point
subsystem for conversion to an equivalent fixed-point representation.
Note If your model contains referenced models, you must run the
Fixed-Point Advisor on each instance of the referenced model as well as
the parent model.
Propose data types based on the simulation results. Examine the results
to resolve any conflicts and to verify that you want to accept the proposed
data type for each result.
Write the proposed data types to the model. Perform the automatic data
typing procedure, which uses the double-precision simulation results to
propose fixed-point data types for appropriately configured blocks. The
Fixed-Point Tool allows you to accept and apply the proposals selectively.
Simulate the model again using the fixed-point settings. Compare the ideal
results for the double-precision run with the fixed-point results.
32-7
32 Convert Floating-Point Model to Fixed Point
32-8
Convert Floating-Point Model to Fixed Point
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_fixed_point_workflow
In this part of the example, you use the Fixed-Point Advisor to prepare the
Controller Subsystem in the ex_fixed_point_workflow model for conversion.
32-9
32 Convert Floating-Point Model to Fixed Point
1 In the Fixed-Point Advisor left pane, expand the Prepare Model for
Conversion folder to view the tasks. For the purpose of this example, run
the tasks in the this folder one at a time. Select Verify model simulation
settings and, in the right pane, select Run this task.
This task validates that model simulation settings allow signal logging and
disables data type override in the model and for fi objects or embedded
numeric data types in your model or workspace. These settings facilitate
conversion to fixed point in later tasks.
A waitbar appears while the task runs. When the run is complete, the
result shows that the task passed.
3 Select and run Address unsupported blocks. This task identifies blocks
that do not support fixed-point data types.
32-10
Convert Floating-Point Model to Fixed Point
The task passes because the subsystem contains no blocks that do not
support fixed-point data.
4 Select and run Set up signal logging. Prior to simulation, you must
specify at least one signal for the Fixed-Point Advisor to use for analysis
and comparison in downstream checks. You should log, at minimum, the
unique input and output signals.
The task generates a warning because signal logging is not specified for
any signals.
The task passes because signal logging is now enabled for at least one
signal.
The Fixed-Point Advisor simulates the model using the current solver
settings, and creates and archives reference signal data in a run named
FPA_Reference to use for analysis and comparison in later conversion
tasks. This task also validates that model simulation settings allow signal
logging and that the Fixed-point instrumentation mode is set to
Minimums, maximums and overflows.
32-11
32 Convert Floating-Point Model to Fixed Point
Running the task after using the Modify All action verifies that you made
the necessary changes. The Analysis Result pane updates to display a
passed result and information about why the task passed.
Tip You can view the reference run data in the Fixed-Point Tool
Contents pane in the run named FPA_Reference or in the Simulation
Data Inspector. Because you ran the simulation twice, the Simulation
Data Inspector displays data for both runs using the same name
(FPA_Reference).
The task passes because none of these options are set to error.
10 Select and run Implement logic signals as Boolean data. This task
verifies that Model Configuration Parameters > Optimization >
32-12
Convert Floating-Point Model to Fixed Point
Note This is a Simulink check. For more information, see “Check bus
usage”.
The task runs and generates a warning because this check works only
from top-level models and you are running from the subsystem. Because
this model uses no buses, ignore this warning. For models containing
buses, you must run the Fixed-Point Advisor from the top-level model to
perform this check.
12 Select and run Simulation range checking. This tasks verifies that the
Model Configuration Parameters > Diagnostics > Simulation range
checking option is not set to none.
The task now passes because the Simulation range checking option is
correct.
15 Select and run Check for implicit signal resolution. This task checks
for models that use implicit signal resolution.
32-13
32 Convert Floating-Point Model to Fixed Point
You have completed all the tasks for the Prepare Model for Conversion
folder. At this point, you can review the results report found at the folder
level, or continue to the next folder.
1 For the purpose of this example, run the tasks in the Prepare for Data
Typing and Scaling folder one at a time.
Open the Prepare for Data Typing and Scaling folder then select
and run Review locked data type settings. This task identifies blocks
that have their data type settings locked down which excludes them for
automatic data typing.
This task passes because the model contains no blocks with locked data
types.
2 Select and run Remove output data type inheritance. This task
identifies blocks that have an inherited output signal data type that might
lead to data type propagation errors.
32-14
Convert Floating-Point Model to Fixed Point
3 In the Fixed-Point Advisor Input Parameters pane, set Data type for
blocks with floating-point inheritance to int16, and rerun the task.
The task fails and the Fixed-Point Advisor provides information about the
failure in the Analysis Result box. The Fixed-Point Advisor recommends
that you set:
• The input data type of the Discrete Filter block, which is a floating-point
inheritance block, to a fixed-point data type to avoid floating-point
inheritance.
• The output data type of all the other blocks that currently have their
output data type set by inheritance rules to the compiled (current
propagated) data type.
4 Fix the failure using the Modify All button to configure the output data
types to the recommended values.
6 Select and run Relax input data type settings. This task identifies
blocks with input data type constraints that might cause data type
propagation issues.
The task passes because the model contains no blocks that have inherited
input data types.
32-15
32 Convert Floating-Point Model to Fixed Point
7 Select and run Verify Stateflow charts have strong data typing with
Simulink. This task verifies that the configuration of all Stateflow charts
ensures strong data typing with Simulink I/O.
The task passes because the configuration of the Stateflow chart in the
subsystem is correct.
The task passes because the model contains no resolved Simulink signal
objects.
9 Select and run Verify hardware selection. This task identifies the
hardware device information in the Hardware Implementation pane of
the Configuration Parameters dialog box. It also checks the default data
type selected for floating-point signals in the model.
The task fails because the default data type for all floating-point signals
is set to Remain floating-point. Because the target hardware is an
embedded processor, the Fixed-Point Advisor recommends that you set this
value to the hardware integer used by the embedded hardware.
10 To fix the failure, in the Input Parameters pane, set Default data type
of all floating-point signals to Same as embedded hardware integer.
12 Select and run Specify block minimum and maximum values. Ideally,
you should specify block output and parameter minimum and maximum
values for, at minimum, the Inport blocks in the system. You can specify
the minimum and maximum values for any block in this step. Typically,
you determine these values during the design process based on the system
you are creating.
The Fixed-Point Advisor warns you that you have not specified any
minimum and maximum values.
32-16
Convert Floating-Point Model to Fixed Point
13 Fix the warning by specifying minimum and maximum values for Inport
blocks:
a Click the Explore Result button.
The Model Advisor Result Explorer opens, showing that the Inport
blocks, In1 and In2, do not have output minimum and maximum values
specified.
b In the center pane, select In1. This block receives the output from
Repeating table Source, which has a minimum value of 10 and a
maximum value of 20. Therefore, set OutMin to 10 and set OutMax
to 20 as follows:
i In the OutMin column for In1, select [] and replace with 10.
ii In the OutMax column for In1, select [] and replace with 20.
c Select In2. This block receives the output from Sine Wave block, which
has a minimum value of -1 and a maximum value of 1. Therefore, set
OutMin to -1 and set OutMax to 1.
d Close the Model Advisor Result Explorer.
The task passes because you specified minimum and maximum values
for all Inport blocks.
The tool advises you to specify minimum and maximum values for all
blocks if possible. For the purpose of this example, do not specify other
minimum and maximum values for other blocks.
You have completed all tasks in the Prepare for Data Typing and Scaling
folder. At this point, you can review the results report found at the folder
level, or continue to the next folder.
32-17
32 Convert Floating-Point Model to Fixed Point
The tool proposes fixed-point data types and scaling based on the ranges of
the Repeating table Source and Sine Wave inputs. You can then use the tool
to accept and apply the proposed data types selectively. In this example, you
propose fraction lengths for the specified word lengths.
The Fixed-Point Tool analyzes the scaling of all fixed-point blocks whose:
• Lock output data type setting against changes by the fixed-point
tools parameter is not selected.
• Output data type parameter specifies a generalized fixed-point
number.
• Data types are not inherited.
2 In the Fixed-Point Tool, set the Column View to Automatic Data Typing
with Simulation Min/Max View to display information relevant to the
proposal. The tool displays the proposed data types in the ProposedDT
column in the Contents pane. The tool does not propose data types for
objects with inherited data types.
3 Examine the results to resolve any conflicts and to ensure that you want
to accept the proposed data type for each result.
32-18
Convert Floating-Point Model to Fixed Point
In the Fixed-Point Tool toolbar, select Show > Conflicts with proposed
data types.
Tip If the tool does detect conflicts, you must resolve these before applying
data types. For more information, see “Examine Results to Resolve
Conflicts” on page 34-20.
Now that you have reviewed the results and ensured that there are no issues,
you are ready to apply the proposed data types to the model, as described in
“Apply Fixed-Point Data Types” on page 32-19.
The Fixed-Point Tool applies the data type proposals to the subsystem
blocks.
The tool has set all the specified data types to the proposed types.
You are now ready to check that the new data types are acceptable, as
described in “Verify Fixed-Point Settings” on page 32-19.
32-19
32 Convert Floating-Point Model to Fixed Point
The Simulink software simulates using the new data types that you
applied in the previous step. Afterward, the Fixed-Point Tool displays
in its Contents pane information about blocks that logged fixed-point
data. The CompiledDT (compiled data type) column for the run shows
that the Controller Subsystem blocks use fixed-point data types with the
new data types.
Tip In the Contents pane, click the Run column heading to sort the runs.
The Fixed-Point Tool plots the signal for the FPA_Reference and
Initial_fixed_point runs, as well as their difference. The difference plot
shows that the floating-point signal and the fixed-point signal are almost
identical, the difference is on the order of 10^-5.
32-20
Convert Floating-Point Model to Fixed Point
Now you are ready to test the fixed-point settings with new the input
data, as described in “Test Fixed-Point Settings With New Input Data” on
page 32-22.
32-21
32 Convert Floating-Point Model to Fixed Point
In this part of the example, you continue working on the model. First, you
modify the range of the Sine Wave input and obtain simulation data based
on this new range. Then, you use the Fixed-Point Tool to refine the model
fixed-point settings based on the new simulation data. The Fixed-Point Tool
proposes new data types that can accommodate the new input range.
To change the range of the input data and test the fixed-point settings:
This action disables data range error detection and allows the simulation
to run to completion.
32-22
Convert Floating-Point Model to Fixed Point
The Fixed-Point Tool Contents pane displays the simulation results for
each block in the subsystem that logged fixed-point data. The tool stores
the results in the Input2 run.
In the Input2 run, the tool highlights in red the result for the Gain block,
indicating that there are issues.
The result shows that the Gain block output saturated, which indicates
that the fixed-point data settings for this block are not suitable for the
new input range.
Next, override the fixed-point data types with doubles and simulate the
model again to obtain the ideal behavior of the subsystem, as described in
“Gather a Floating-Point Benchmark” on page 32-23.
2 In the Settings for selected system pane, set Data type override to
Double.
Using this setting, the Fixed-Point Tool performs a global override of the
fixed-point data types and scaling using double-precision numbers, thus
avoiding quantization effects.
32-23
32 Convert Floating-Point Model to Fixed Point
The Fixed-Point Tool highlights any simulation results that have issues,
such as overflows or saturations.
5 In the Contents pane, click the Run column to sort the runs. Verify that
there were no overflows or saturations in the DTO_Input2 run.
2 In the Propose Data Types dialog box, select DTO_Input2 as the run to
use for proposing data types, and then click OK.
The Fixed-Point Tool proposes new data types for all objects in the model
and updates the results in the Contents pane.
3 In the Fixed-Point Tool, set the Column View to Automatic Data Typing
with Simulation Min/Max View to display information relevant to the
proposal. The tool displays the proposed data types in the ProposedDT
column in the Contents pane. The tool does not propose data types for
objects with inherited data types.
4 Examine the results to resolve any conflicts and to ensure that you want
to accept the proposed data type for each result.
In the Fixed-Point Tool toolbar, select Show > Conflicts with proposed
data types.
The Fixed-Point Tool detected no conflicts, so you are ready to apply the
new data types as described in “Apply the New Fixed-Point Data Types” on
page 32-25.
32-24
Convert Floating-Point Model to Fixed Point
2 In the Apply Data Types dialog box, select DTO_Input2 as the run to use
for applying proposed data types and then click OK.
The tool has set all the specified data types to the proposed types.
2 In the Settings for selected system pane, set Data type override to
Use local settings.
The Simulink software simulates using the new data types that you applied
in the previous step and stores the results in the Final_fixed_point run.
32-25
32 Convert Floating-Point Model to Fixed Point
The Fixed-Point Tool plots the signal for both runs, as well as their
difference. The difference plot shows that the floating-point signal and
the fixed-point signal are identical.
8 Optionally, you can zoom in to view the steady-state region with greater
detail. From the Tools menu of the figure window, select Zoom In and
then drag the pointer to draw a box around the area you want to view
more closely.
The Model Advisor warns that your model contains expensive rounding
code.
32-26
Convert Floating-Point Model to Fixed Point
8 The Discrete Filter and Gain blocks contain expensive multiplication code.
9 The Model Advisor recommends that you change the breakpoints data of
your lookup tables to have even power of 2 spacing, and use Evenly-spaced
points index search method.
The Model Advisor warns that the model contains expensive saturation
code.
12 Clear the Gain block’s Function Block Parameters > Signal Attributes
> Saturate on integer overflow parameter. Click apply, and close the
window.
32-27
32 Convert Floating-Point Model to Fixed Point
• Ensure that you simulate the system using the full range of inputs.
If you use simulation minimum and maximum values to scale fixed-point
data types, the tools provide meaningful results when exercising the full
range of values over which your design is meant to run.
32-28
Where to Learn More
32-29
32 Convert Floating-Point Model to Fixed Point
32-30
33
In creating the points for a function lookup table, you generally want to
achieve one or both of the following goals:
“Create Lookup Tables for a Sine Function” on page 33-6 shows you how to
create function lookup tables using the function fixpt_look1_func_approx.
You can optimize the lookup table to minimize the number of data points, the
error, or both. You can also restrict the spacing of the breakpoints to be even
or even powers of two to speed up computations using the table.
“Worst-Case Error for a Lookup Table” on page 33-3 explains how to use the
function fixpt_look1_func_plot to find the worst-case error of a lookup
table and plot the errors at all points.
33-2
Worst-Case Error for a Lookup Table
For example, if the ideal function is the square root, and the breakpoints of
the lookup table are 0, 0.25, and 1, then in a perfect implementation of the
lookup table, the worst-case error is 1/8 = 0.125, which occurs at the point 1/16
= 0.0625. In practice, the error could be greater, depending on the fixed-point
quantization and other factors.
33-3
33 Producing Lookup Table Data
To get the worst-case error of the lookup table and a plot of the error, type:
errworst =
0.1250
33-4
Worst-Case Error for a Lookup Table
The upper box (Outputs) displays a plot of the square root function with a
plot of the fixed-point lookup approximation underneath. The approximation
is found by linear interpolation between the breakpoints. The lower box
(Absolute Error) displays the errors at all points in the interval from 0 to 1.
Notice that the maximum absolute error occurs at 0.0625. The error at the
breakpoints is 0.
33-5
33 Producing Lookup Table Data
Introduction
The sections that follow explain how to use the function
fixpt_look1_func_approx to create lookup tables. It gives
examples that show how to create lookup tables for the function sin(2πx) on
the interval from 0 to 0.25.
33-6
Create Lookup Tables for a Sine Function
You must use at least one of the parameters errmax and nptsmax. The next
section, “Setting Function Parameters for the Lookup Table” on page 33-8,
gives typical settings for these parameters.
The section “Specifying Both errmax and nptsmax” on page 33-20 describes
how the function behaves when you specify both errmax and nptsmax.
Spacing
You can use the optional spacing parameter to restrict the spacing between
breakpoints of the lookup table. The options are
• 'unrestricted' — Default.
• 'even' — Distance between any two adjacent breakpoints is the same.
• 'pow2' — Distance between any two adjacent breakpoints is the same
and the distance is a power of two.
33-7
33 Producing Lookup Table Data
The section “Restricting the Spacing” on page 33-13 and the examples that
follow it explain how to use the spacing parameter.
If you exit the MATLAB software after typing these commands, you must
retype them before trying any of the other examples in this section.
errmax = 2^-10;
33-8
Create Lookup Tables for a Sine Function
xmin,xmax,xdt,xscale,ydt,yscale,rndmeth,errmax,[]);
Note that the nptsmax and spacing parameters are not specified.
length(xdata)
ans =
16
This means that 16 points are required to approximate sin(2πx) to within the
tolerance specified by errmax.
You can display the maximum error by typing errworst. This returns
errworst
errworst =
9.7656e-04
33-9
33 Producing Lookup Table Data
fixpt_look1_func_plot(xdata,ydata,funcstr,xmin,xmax,xdt, ...
xscale,ydt,yscale,rndmeth);
The upper plot shows the ideal function sin(2πx) and the fixed-point lookup
approximation between the breakpoints. In this example, the ideal function
and the approximation are so close together that the two graphs appear to
coincide. The lower plot displays the errors.
33-10
Create Lookup Tables for a Sine Function
to the points in xdata. However, you can define a different set of values for
ydata after running fixpt_look1_func_plot. This can sometimes reduce
the maximum error.
You can also change the values of xmin and xmax in order to evaluate the
lookup table on a subset of the original interval.
To find the new maximum error after changing ydata, xmin or xmax, type
nptsmax = 21;
The empty brackets, [], tell the function to ignore the parameter errmax,
which is not used in this example. Omitting errmax causes the function
fixpt_look1_func_approx to return a lookup table of size specified by
nptsmax, with the smallest worst-case error.
The function returns a vector xdata with 21 points. You can find the
maximum error for this set of points by typing errworst at the MATLAB
prompt.
33-11
33 Producing Lookup Table Data
errworst
errworst =
5.1139e-04
fixpt_look1_func_plot(xdata,ydata,funcstr,xmin,xmax,xdt, ...
xscale,ydt,yscale,rndmeth);
33-12
Create Lookup Tables for a Sine Function
• 'unrestricted' — Default.
33-13
33 Producing Lookup Table Data
Both power of two and even spacing increase the computational speed of
the lookup table and use less command read-only memory (ROM). However,
specifying either of the spacing restrictions along with errmax usually
requires more data points in the lookup table than does unrestricted spacing
to achieve the same degree of accuracy. The section “Effects of Spacing on
Speed, Error, and Memory Usage” on page 33-24 discusses the tradeoffs
between different spacing options.
spacing = 'even';
[xdata ydata errworst] = fixpt_look1_func_approx(funcstr, ...
xmin,xmax,xdt,xscale,ydt,yscale,rndmeth,errmax,[],spacing);
You can find the number of points in the lookup table by typing:
length(xdata)
ans =
20
fixpt_look1_func_plot(xdata,ydata,funcstr,xmin,xmax,xdt, ...
xscale,ydt,yscale,rndmeth);
33-14
Create Lookup Tables for a Sine Function
33-15
33 Producing Lookup Table Data
spacing = 'even';
[xdata ydata errworst] = fixpt_look1_func_approx(funcstr, ...
xmin,xmax,xdt,xscale,ydt,yscale,rndmeth,[],nptsmax,spacing);
fixpt_look1_func_plot(xdata,ydata,funcstr,xmin,xmax,xdt, ...
xscale,ydt,yscale,rndmeth);
33-16
Create Lookup Tables for a Sine Function
spacing = 'pow2';
[xdata ydata errworst] = ...
fixpt_look1_func_approx(funcstr,xmin, ...
xmax,xdt,xscale,ydt,yscale,rndmeth,errmax,[],spacing);
To find out how many points are in the lookup table, type
length(xdata)
ans =
33
This means that 33 points are required to achieve the worst-case error
specified by errmax. To verify that these points are evenly spaced, type
widths = diff(xdata)
This generates a vector whose entries are the differences between consecutive
points in xdata. Every entry of widths is 2-7.
errworst
errworst =
3.7209e-04
33-17
33 Producing Lookup Table Data
To plot the lookup table data along with the errors, type
fixpt_look1_func_plot(xdata,ydata,funcstr,xmin,xmax,xdt, ...
xscale,ydt,yscale,rndmeth);
33-18
Create Lookup Tables for a Sine Function
spacing = 'pow2';
[xdata ydata errworst] = ...
fixpt_look1_func_approx(funcstr,xmin, ...
xmax,xdt,xscale,ydt,yscale,rndmeth,errmax,[],spacing);
fixpt_look1_func_plot(xdata,ydata,funcstr,xmin,xmax,xdt, ...
xscale,ydt,yscale,rndmeth);
33-19
33 Producing Lookup Table Data
1 Power of two
33-20
Create Lookup Tables for a Sine Function
2 Even
3 Unrestricted
If the function cannot find any lookup table satisfying both conditions, it
ignores nptsmax and returns a lookup table with unrestricted spacing, whose
worst-case error is at most errmax. In this case, the function behaves the
same as if the nptsmax parameter were omitted.
The results for three different settings for nptsmax are as follows:
• nptsmax = 33; — The function creates the lookup table with 33 points
having power of two spacing, as in Example 3.
• nptsmax = 21; — Because the errmax and nptsmax conditions cannot be
met with power of two spacing, the function creates the lookup table with
20 points having even spacing, as in Example 5.
• nptsmax = 16; — Because the errmax and nptsmax conditions cannot be
met with either power of two or even spacing, the function creates the
lookup table with 16 points having unrestricted spacing, as in Example 1.
Worst-Case Number of
Example Options Spacing Error Points in Table
1 errmax=2^-10 'unrestricted' 2^-10 16
2 nptsmax=21 'unrestricted' 2^-10.933 21
3 errmax=2^-10 'even' 2^-10.0844 20
33-21
33 Producing Lookup Table Data
Worst-Case Number of
Example Options Spacing Error Points in Table
4 nptsmax=21 'even' 2^-10.2209 21
5 errmax=2^-10 'pow2' 2^-11.3921 33
6 nptsmax=21 'pow2' 2^-9.627 17
33-22
Use Lookup Table Approximation Functions
1 Define:
4 Vary input criteria, such as errmax, nptsmax, and spacing, to produce sets
of X and Y data that generate functions with varying worst-case error,
number of points required, and spacing.
33-23
33 Producing Lookup Table Data
This comparison is valid only when the breakpoints are not tunable. If the
breakpoints are tunable in the generated code, all three cases generate the
same code. For a summary of the effects of breakpoint spacing on execution
speed, error, and memory usage, see “Summary of the Effects of Breakpoint
Spacing” on page 33-31.
33-24
Effects of Spacing on Speed, Error, and Memory Usage
fxpdemo_approx_sin
Uneven Case
Uneven spacing requires both Y data points and breakpoints:
int16_T yuneven[8];
uint16_T xuneven[8];
Even Case
Even spacing requires only Y data points:
int16_T yeven[10];
The total bytes used is 20. The breakpoints are not explicitly required. The
code uses the spacing between the breakpoints, and might use the smallest
and largest breakpoints. At most, three values related to the breakpoints
are necessary.
int16_T ypow2[17];
33-25
33 Producing Lookup Table Data
The total bytes used is 34. The breakpoints are not explicitly required. The
code uses the spacing between the breakpoints, and might use the smallest
and largest breakpoints. At most, three values related to the breakpoints
are necessary.
Uneven Case
Unevenly-spaced breakpoints require a general-purpose algorithm such as a
binary search to determine where the input lies in relation to the breakpoints.
The following code provides an example:
iLeft = 0;
iRght = 7; /* number of breakpoints minus 1 */
33-26
Effects of Spacing on Speed, Error, and Memory Usage
Even Case
Evenly-spaced breakpoints require only one step to determine where the
input lies in relation to the breakpoints:
The divisor 455U represents the spacing between breakpoints. In general, the
dividend would be (uAngle - SmallestBreakPoint). In this example, the
smallest breakpoint is zero, so the code optimizes out the subtraction.
The number of shifts is 8 because the breakpoints have spacing 2^8. The
smallest breakpoint is zero, so uAngle replaces the general case of (uAngle -
SmallestBreakPoint).
Comparison
To determine where the input lies with respect to the breakpoints, the
unevenly-spaced case requires much more code than the other two cases.
This code requires additional command ROM. If many lookup tables share
the binary search algorithm as a function, you can reduce this ROM penalty.
Even if the code is shared, the number of clock cycles required to determine
the location of the input is much higher for the unevenly-spaced cases than
the other two cases. If the code is shared, function-call overhead decreases
the speed of execution a little more.
In the evenly-spaced case and the power of two spaced case, you can determine
the location of the input with a single line of code. The evenly-spaced case
uses a general integer division. The power of two case uses a shift instead
of general division because the divisor is an exact power of two. Without
33-27
33 Producing Lookup Table Data
knowing the specific processor, you cannot be certain that a shift is better
than division.
Most processors provide a way to do logical and arithmetic shifts left and
right. A key difference is whether the processor can do N shifts in one
instruction (barrel shift) or requires N instructions that shift one bit at a time.
The barrel shift requires less code. Whether the barrel shift also increases
speed depends on the hardware that supports the operation.
The compiler can also complicate the comparison. In the previous example,
the command uAngle >> 8 essentially takes the upper 8 bits in a 16-bit word.
The compiler can detect this situation and replace the bit shifts with an
instruction that takes the bits directly. If the number of shifts is some other
value, such as 7, this optimization would not occur.
Uneven Case
For the uneven case, one possible implementation of the ideal interpolation
in fixed point is as follows:
33-28
Effects of Spacing on Speed, Error, and Memory Usage
The multiplication and division routines are not shown here. These routines
can be complex and depend on the target processor. For example, these
routines look different for a 16-bit processor than for a 32-bit processor.
Even Case
Evenly-spaced breakpoints implement interpolation using slightly different
calculations than the uneven case. The key difference is that the calculations
do not directly use the breakpoints. When the breakpoints are not required in
ROM, you can save a lot of memory:
MUL_S16_U16_S16_SR8(yPow2,lambda,yPow2);
33-29
33 Producing Lookup Table Data
yPow2 += ypow2[iLeft];
This implementation has significant advantages over the uneven and even
implementations:
• A bitwise AND combined with a shift right at the end of the multiplication
replaces a subtraction and a division.
• The term (u - xData[iLeft] ) / ( xData[iRght] - xData[iLeft])
results in no loss of precision, because the spacing is a power of two.
In contrast, the uneven and even cases usually introduce rounding error in
this calculation.
33-30
Effects of Spacing on Speed, Error, and Memory Usage
The number of Y data points follows the expected pattern. For the same
worst-case error, unrestricted spacing (uneven) requires the fewest data
points, and power-of-two-spaced breakpoints require the most. However, the
implementation for the evenly-spaced and the power of two cases does not
33-31
33 Producing Lookup Table Data
need the breakpoints in the generated code. This reduces their data ROM
requirements by half. As a result, the evenly-spaced case actually uses less
data ROM than the unevenly-spaced case. Also, the power of two case requires
only slightly more ROM than the uneven case. Changing the worst-case error
can change these rankings. Nonetheless, when you compare data ROM usage,
you should always take into account the fact that the evenly-spaced and power
of two spaced cases do not require their breakpoints in ROM.
The effort of calculating the interpolation favors the power of two case, which
uses a bitwise AND operation and a shift to replace a subtraction and a
division. The advantage of this behavior depends on the specific hardware,
but you can expect an advantage in code size, speed, and also in accuracy. The
evenly-spaced case calculates the interpolation with a minor improvement in
efficiency over the unevenly-spaced case.
33-32
34
You can use the Fixed-Point Tool to select data types automatically for your
model using the following methods.
34-2
Before Proposing Data Types for Your Model
• The Fixed-Point Tool proposes new data types for the fixed-point data
types in your model. If you choose to apply the proposed data types, the
tool changes the data types in your model. Before using the Fixed-Point
Tool, back up your model and workspace variables to ensure that you
can recover your original data type settings and capture the fixed-point
instrumentation and data type override settings using the Shortcut Editor.
For more information, see “Best Practices for Fixed-Point Workflow” on
page 34-5.
• Before proposing data types, verify that you can update diagram
successfully . Sometimes, changing the data types in your model results
in subsequent update diagram errors. Immediately before and after
applying data type proposals, it is good practice to test update diagram
again. This practice enables you to fix any errors before making further
modifications to your model.
For more information, see “Update a Block Diagram”.
• The Fixed-Point Tool alerts you to potential issues with proposed data
types for each object in your model:
- If the Fixed-Point Tool detects that the proposed data type introduces
data type errors when applied to an object, the tool marks the object
with an error, . You must inspect this proposal and fix the problem in
the Simulink model. After fixing the problem, rerun the simulation and
generate a proposal again to confirm that you have resolved the issue.
For more information, see “Examine Results to Resolve Conflicts” on
page 34-20.
- If the Fixed-Point Tool detects that the proposed data type poses
potential issues for an object, the tool marks the object with a yellow
caution, . Review the proposal before accepting it.
- If the Fixed-Point Tool detects that the proposed data type poses no
issues for an object, the tool marks the object with a green check, .
34-3
34 Automatic Data Typing
Caution The Fixed-Point Tool does not detect all potential data type
issues. If the Fixed-Point Tool does not detect any issues for your model,
it is still possible to experience subsequent data type propagation
issues. For more information, see “Models That Might Cause Data Type
Propagation Errors” on page 34-8.
34-4
Best Practices for Fixed-Point Workflow
34-5
34 Automatic Data Typing
• You already know the fixed-point data types that you want to use for a
particular block.
For example, the block is modeling a real-world component. Set up the
block to allow for known hardware limitations, such as restricting outputs
to integer values.
Explicitly specify the output data type of the block and select Lock output
data type setting against changes by the fixed-point tools.
• You are debugging a model and know that a particular block accepts only
certain input signal data types.
Explicitly specify the output data type of upstream blocks and select Lock
output data type setting against changes by the fixed-point tools.
34-6
Best Practices for Fixed-Point Workflow
• Use the failure information to fix the errors in your model. After fixing the
errors, test update diagram again.
• If you are unable to fix the errors, restore your back-up model. After
restoring the model, try to fix the errors by, for example, locking output
data type settings and isolating the system, as described in the preceding
sections. After addressing the errors, test update diagram again.
34-7
34 Automatic Data Typing
The Fixed-Point Tool does not detect all potential data type issues. If the tool
does not report any issues for your model, it is still possible to experience
subsequent data type propagation errors. Before you use the Fixed-Point
Tool, back up your model to ensure that you can recover your original data
type settings. For more information, see “Best Practices for Fixed-Point
Workflow” on page 34-5.
The following models are likely to cause data type propagation issues.
34-8
Models That Might Cause Data Type Propagation Errors
34-9
34 Automatic Data Typing
34-10
Models That Might Cause Data Type Propagation Errors
34-11
34 Automatic Data Typing
34-12
Automatic Data Typing Using Simulation Data
34-13
34 Automatic Data Typing
1 Back up your model in case of error and as a baseline for testing and
validation.
3 From the Simulink menu, select Simulation > Mode > Normal so that
the model runs in Normal mode. The Fixed-Point Tool supports only
Normal mode.
4 If you are using design minimum and maximum range information, add
this information to blocks.
You specify a design range for model objects using parameters typically
titled Output minimum and Output maximum. For a list of blocks
that permit you to specify these values, see “Blocks That Allow Signal
Range Specification”.
5 Specify fixed-point data types for blocks and signals in your model. For
blocks with the Data Type Assistant, use the Calculate Best-Precision
Scaling button to calculate best-precision scaling automatically. For more
information, see “Specify Fixed-Point Data Types with the Data Type
Assistant” on page 26-24. Use the Fixed-Point Advisor to prepare your
34-14
Automatic Data Typing Using Simulation Data
6 You can choose to lock some blocks against automatic data typing by
selecting the Lock output data type setting against changes by the
fixed-point tools parameter. If you select the Lock output data type
setting against changes by the fixed-point tools parameter, the tool
does not propose data types for that object.
If update diagram fails, use the failure information to fix the errors in
your model. After fixing the errors, test update diagram again. If you are
unable to fix the errors, restore your back-up model.
8 If the model changed, back up the model again in case of error and as a
baseline for testing and validation.
34-15
34 Automatic Data Typing
Use the Fixed-Point Advisor to prepare the model for conversion. For more
information, see “Preparation for Fixed-Point Conversion” on page 30-2.
34-16
Automatic Data Typing Using Simulation Data
Note You can plot results only for signals that have signal logging enabled.
Note Data type override does not apply to boolean or enumerated data
types.
The Fixed-Point Tool highlights any simulation results that have issues,
such as overflows or saturations.
34-17
34 Automatic Data Typing
When proposing data types, the Fixed-Point Tool collects the following types
of range data for model objects:
The Fixed-Point Tool uses available range data to calculate data type
proposals according to the following rules:
34-18
Automatic Data Typing Using Simulation Data
If these options are not visible, use the Configure link to display them.
Note When the Fixed-Point Tool proposes data types, it does not alter
your model.
If there are conflicts in your model, the Fixed-Point Tool displays the
Result Details dialog box.
34-19
34 Automatic Data Typing
If you do not see this warning, there are no conflicts in your model. Go to
“Apply Proposed Data Types” on page 34-24.
1 On the Contents pane, select an object that has proposed data types.
The Result Details dialog box provides the following information about the
proposed data types, as appropriate.
Summary
Details which run the result is in and the current data type specified for
the selected object.
Needs Attention
Lists potential issues and errors associated with data type proposals. It
describes the issues and suggests methods for resolving them. The dialog box
uses the following icons to differentiate warnings from errors:
34-20
Automatic Data Typing Using Simulation Data
The dialog box provides a hyperlink that you can click to highlight the objects
that share data types in the model. To clear this highlighting, from the model
View menu, select Remove Highlighting.
The Fixed-Point Tool allocates an identification tag to objects that must share
the same data type. The tool displays this identification tag in the DTGroup
column for the object. To display only the objects that must share data types,
from the Fixed-Point Tool main toolbar, select the Show option.
Item Description
34-21
34 Automatic Data Typing
Item Description
The dialog box table also includes a column titled Percent Proposed
Representable. This column indicates the percentage of the proposed
representable range that each value covers. Overflows occur when values
lie outside this range.
Shared Values. When proposing data types, the Fixed-Point Tool attempts
to satisfy data type requirements that model objects impose on one another.
For example, the Sum block provides an option that requires all of its inputs
to have the same data type. Consequently, the dialog box table might also
list attributes of other model objects that impact the proposal for the selected
object. In such cases, the table displays the following types of shared values:
• Initial Values
Some model objects provide parameters that allow you to specify the
initial values of their signals. For example, the Constant block includes
a Constant value parameter that initializes the block output signal.
The Fixed-Point Tool uses initial values to propose data types for model
objects whose design and simulation ranges are unavailable. When data
34-22
Automatic Data Typing Using Simulation Data
type dependencies exist, the tool considers how initial values impact the
proposals for neighboring objects.
• Model-Required Parameters
Some model objects require the specification of numeric parameters to
compute the value of their outputs. For example, the Table data parameter
of an n-D Lookup Table block specifies values that the block requires to
perform a lookup operation and generate output. When proposing data
types, the Fixed-Point Tool considers how this “model-required” parameter
value impacts the proposals for neighboring objects.
The Fixed-Point Tool lists its data type proposals on the Contents pane
under the ProposedDT column. The tool alerts you to potential issues for
each object in the list by displaying a green, yellow, or red icon.
The proposed data type will introduce data type errors if applied
to this object.
b Click the Show details for selected result button to open the
Result Details dialog box.
c Use the information provided in the Needs Attention section of the
Result Details dialog box to resolve the conflict by fixing the problem in
the Simulink model.
3 Review the Result Details for the warnings and correct the problem
if necessary.
34-23
34 Automatic Data Typing
4 You have changed the Simulink model, so the benchmark data is not up to
date. Click the Fixed-Point Tool Start button to rerun the simulation.
The Fixed-Point Tool warns you that you have not applied proposals. Click
the Ignore and Simulate button to continue.
6 On the Fixed-Point Tool toolbar, use the Show option to filter the results
to show All results.
The Fixed-Point Tool will apply the proposed data type to this object.
By default, the tool selects the Accept check box when a proposal
differs from the object’s current data type.
The Fixed-Point Tool will ignore the proposed data type and leave the
current data type intact for this object.
No proposal exists for this object, for example, when the object
specifies a data type inheritance rule or is locked against automatic
data typing.
1 Examine each result. For more information about a particular result, select
the result and then click the Show details for selected result button
to display the Result Details dialog box.
2 If you do not want to accept the proposal for a result, on the Fixed-Point
Tool Contents pane, clear the Accept check box for that result.
Before applying proposals to your model, you can customize them with the
Fixed-Point Tool. On the Contents pane, click a ProposedDT cell and
34-24
Automatic Data Typing Using Simulation Data
edit the data type expression. For information about specifying fixed-point
data types, see fixdt.
If you have not fixed all the warnings in the model, the Fixed-Point Tool
displays a warning dialog box.
This sets:
• Data type override to Use local settings.
• Fixed-point instrumentation mode to Minimums, maximums and
overflows.
• The run name (in the Data collection pane Store results in run field)
to NoOverride.
Using these settings, the Fixed-Point Tool simulates the model using the
new fixed-point settings and logs minimum value, maximum value, and
overflow data for all blocks in the current system or subsystem in the run
NoOverride.
4 Compare the ideal results stored in the DoubleOverride run with the
fixed-point results in the NoOverride run:
a On the Contents pane, select a result that has logged signal data. These
results are annotated with the icon.
34-25
34 Automatic Data Typing
If you have more than two runs, in the Compare Runs Selector dialog
box, select the two runs that you want to compare.
Caution The Fixed-Point Tool does not save the changes to the signal object.
Before closing the model, you must save the changes.
After automatic data typing, if you delete or manipulate a signal object in the
base workspace, you must rerun the automatic data typing.
34-26
Automatic Data Typing Using Derived Ranges
Range analysis:
2 Set up model.
34-27
34 Automatic Data Typing
10 Update diagram.
1 Back up your model in case of error and as a baseline for testing and
validation.
3 Select Simulation > Normal in the Simulink menu so that the model runs
in Normal mode. The Fixed-Point Tool supports only Normal mode.
4 To autoscale using derived data, you must specify design minimum and
maximum values on at least the model inputs. The range analysis tries to
narrow the derived range by using all the specified design ranges in the
model. The more design range information you specify, the more likely
the range analysis is to succeed. As the analysis is performed, it derives
new range information for the model and then attempts to use this new
information together with the specified ranges to derive ranges for the
remaining objects in the model. For this reason, the analysis results might
depend on block priorities because these priorities determine the order in
which the software analyzes the blocks.
You specify a design range for model objects using parameters typically
titled Output minimum and Output maximum. For a list of blocks
34-28
Automatic Data Typing Using Derived Ranges
that permit you to specify these values, see “Blocks That Allow Signal
Range Specification”.
5 Specify fixed-point data types for blocks and signals in your model. For
blocks with the Data Type Assistant, use the Calculate Best-Precision
Scaling button to calculate best-precision scaling automatically. For more
information, see “Specify Fixed-Point Data Types with the Data Type
Assistant” on page 26-24.
6 You can choose to lock some blocks against automatic data typing by
selecting the Lock output data type setting against changes by the
fixed-point tools parameter. If you select the Lock output data type
setting against changes by the fixed-point tools parameter, the tool
does not propose data types for that object.
If update diagram fails, use the failure information to fix the errors in
your model. After fixing the errors, test update diagram again. If you are
unable to fix the errors, restore your back-up model.
8 If the model changed, back up the model in case of error and as a baseline
for testing and validation.
34-29
34 Automatic Data Typing
Use the Fixed-Point Advisor to prepare the model for conversion. For more
information, see “Preparation for Fixed-Point Conversion” on page 30-2.
2 On the Settings for selected system pane, set Data type override to
Double.
34-30
Automatic Data Typing Using Derived Ranges
The analysis runs and tries to derive range information for objects in the
selected system.
If the analysis successfully derives range data for the model, the Fixed-Point
Tool displays the derived minimum and maximum values for the blocks in
the selected system. (See “View Derived Ranges in the Fixed-Point Tool” on
page 35-12.) Before proposing data types, review the results.
If the analysis fails, examine the error messages and resolve the issues.
See “Resolve Range Analysis Issues” on page 34-32.
34-31
34 Automatic Data Typing
34-32
Automatic Data Typing Using Derived Ranges
When generating data type proposals, the Fixed-Point Tool collects the
following types of range data for model objects:
The Fixed-Point Tool uses available range data to calculate data type
proposals according to the following rules:
34-33
34 Automatic Data Typing
• The tool observes the derived range only when the Derived min/max
option is selected. Otherwise, the tool ignores the derived range.
The Safety margin for design and derived min/max (%) parameter
specifies a range that differs from that defined by the derived range.
For more information, see Percent safety margin for design and derived
min/max in the fxptdlg reference.
• The tool observes the simulation range only when the Simulation
min/max option is selected. Otherwise, the tool ignores the simulation
range.
The Safety margin for simulation min/max (%) parameter specifies a
range that differs from that defined by the simulation range. For more
information, see “Safety margin for simulation min/max (%)” in the
fxptdlg reference.
If these options are not visible, use the Configure link to display them.
2 If you have a safety margin to apply, set Safety margin for design and
derived min/max (%). For example, enter 10 for a 10% safety margin.
Note When the Fixed-Point Tool proposes data types, it does not alter
your model.
If there are conflicts in your model, the Fixed-Point Tool displays the
Result Details dialog box.
If you do not see this warning, there are no conflicts in your model, go to
“Apply Proposed Data Types” on page 34-24.
34-34
Automatic Data Typing Using Derived Ranges
1 On the Contents pane, select an object that has proposed data types.
34-35
34 Automatic Data Typing
34-36
Automatic Data Typing Using Derived Ranges
The Result Details dialog box provides the following information about the
proposed data type, as appropriate.
Summary
Details about which run the result is in and the current data type specified
for the selected object.
Needs Attention
Lists potential issues and errors associated with data type proposals.
Describes the issues and suggests methods for resolving them. The dialog box
uses the following icons to differentiate warnings from errors.
**The Shared Data Type Summary also reports all source blocks that drive
the same element of a bus object. **
The dialog box provides a hyperlink that you can click to highlight the objects
that share data types in the model. To clear this highlighting, from the model
View menu, select Remove Highlighting.
34-37
34 Automatic Data Typing
The Fixed-Point Tool allocates an identification tag to objects that must share
the same data type. The tool displays this identification tag in the DTGroup
column for the object. To display only the objects that must share data types,
from the Fixed-Point Tool main toolbar, select the Show option.
Item Description
34-38
Automatic Data Typing Using Derived Ranges
Item Description
The dialog box table also includes a column titled Percent Proposed
Representable. This column indicates the percentage of the proposed
representable range that each value covers. Overflows occur when values
lie outside this range.
Shared Values. When proposing data types, the Fixed-Point Tool attempts
to satisfy data type requirements that model objects impose on one another.
For example, the Sum block provides an option that requires all of its inputs
to have the same data type. Consequently, the dialog box table might also
list attributes of other model objects that impact the data type proposal for
the selected object. In such cases, the table displays the following types
of shared values:
• Initial Values
Some model objects provide parameters that allow you to specify the
initial values of their signals. For example, the Constant block includes
a Constant value parameter that initializes the block output signal.
The Fixed-Point Tool uses initial values to propose data types for model
objects whose design and simulation ranges are unavailable. When data
type dependencies exist, the tool considers how initial values impact the
proposals for neighboring objects.
• Model-Required Parameters
34-39
34 Automatic Data Typing
The Fixed-Point Tool lists its data type proposals on the Contents pane
under the ProposedDT column. The tool alerts you to potential issues for
each object in the list by displaying a green, yellow, or red icon.
The proposed data type will introduce data type errors if applied
to this object.
b Click the Show details for selected result button to open the
Result Details dialog box.
c Use the information provided in the Needs Attention section of the
Result Details dialog box to resolve the conflict by fixing the problem in
the Simulink model.
3 Review the Result Details for the warnings and correct the problem
if necessary.
4 You have changed the Simulink model, so the benchmark data is not up to
date. Click the Fixed-Point Tool Start button to rerun the simulation.
34-40
Automatic Data Typing Using Derived Ranges
The Fixed-Point Tool warns you that you have not applied proposals. Click
the Ignore and Simulate button to continue.
6 On the Fixed-Point Tool toolbar, use the Show option to filter the results
to show All results.
The Fixed-Point Tool will apply the proposed data type to this object.
By default, the tool selects the Accept check box when a proposal
differs from the object’s current data type.
The Fixed-Point Tool will ignore the proposed data type and leave the
current data type intact for this object.
No proposal exists for this object, for example, when the object
specifies a data type inheritance rule or is locked against automatic
data typing.
1 Examine each result. For more information about a particular result, select
the result and then click the Show details for selected result button
to open the Result Details dialog box.
2 If you do not want to accept the proposal for a result, on the Fixed-Point
Tool Contents pane, clear the Accept check box for that result.
Before applying proposals to your model, the Fixed-Point Tool enables you
to customize them. On the Contents pane, click a ProposedDT cell and
edit the data type expression. For information about specifying fixed-point
data types, see fixdt.
34-41
34 Automatic Data Typing
If you have not fixed all the warnings in the model, the Fixed-Point Tool
displays a warning dialog box.
Update Diagram
From the model’s Simulation menu, select Update Diagram.
After applying the data types to the model, update diagram to check for
data type propagation issues.
If update diagram fails, use the failure information to fix the errors in your
model. After fixing the errors, test update diagram again. If you are unable
to fix the errors, restore your backed up model.
34-42
Propose Fraction Lengths
3 If you have safety margins to apply, set Safety margin for design and
derived min/max (%) and Safety margin for design and derived
min/max (%), as applicable.
Note When the Fixed-Point Tool proposes data types, it does not alter
your model.
If there are conflicts in your model, the Fixed-Point Tool opens the Result
Details dialog box.
If you do not see this warning, there are no conflicts in your model. Review
the proposed word lengths,
34-43
34 Automatic Data Typing
The Simulink model of the feedback design consists of the following blocks
and subsystems:
• Reference
34-44
Propose Fraction Lengths
Simulation Setup
To set up this kind of fixed-point feedback controller simulation:
34-45
34 Automatic Data Typing
In the real world, there are design components with fixed characteristics
(the hardware) and design components with characteristics that you
can change (the software). In this feedback design, the main hardware
components are the A/D hardware, the D/A hardware, and the analog
plant. The main software component is the digital controller.
You evaluate the behavior of the plant and the controller with a Bode plot.
This evaluation is idealized, because all numbers, operations, and states
are double-precision.
34-46
Propose Fraction Lengths
0
Magnitude
10
−5
10
−1 0 1 2 3
10 10 10 10 10
Freq (rad/sec)
−100
Phase
−200
−300
−400
−1 0 1 2 3
10 10 10 10 10
Freq (rad/sec)
The Bode plots were created using workspace variables produced by a script
named preload_feedback.m.
34-47
34 Automatic Data Typing
• Up Cast
34-48
Propose Fraction Lengths
Up Cast is a Data Type Conversion block that connects the A/D hardware
with the digital controller. It pads the output word size of the A/D hardware
with trailing zeros to a 16-bit number (the base data type).
• Numerator Terms and Denominator Terms
Each of these Discrete FIR Filter blocks represents a weighted sum carried
out in the CPU target. The word size and precision in the calculations
reflect those of the accumulator. Numerator Terms multiplies and
accumulates the most recent inputs with the FIR numerator coefficients.
Denominator Terms multiples and accumulates the most recent delayed
outputs with the FIR denominator coefficients. The coefficients are stored
in ROM using the base data type. The most recent inputs are stored in
global RAM using the base data type.
• Combine Terms
Combine Terms is a Sum block that represents the accumulator in the
CPU. Its word size and precision are twice that of the RAM (double bits).
• Down Cast
Down Cast is a Data Type Conversion block that represents taking the
number from the CPU and storing it in RAM. The word size and precision
are reduced to half that of the accumulator when converted back to the
base data type.
• Prev Out
Prev Out is a Unit Delay block that delays the feedback signal in memory
by one sample period. The signals are stored in global RAM using the
base data type.
N N
y ( k) = ∑ biu ( k − 1) − ∑ ai y ( k − 1) ,
i =0 i=1
34-49
34 Automatic Data Typing
This example shows you how to use the Fixed-Point Tool to refine the scaling
of fixed-point data types associated with a feedback controller model (see
“About the Feedback Controller Example Model” on page 34-44). Although the
tool enables multiple workflows for converting a digital controller described in
ideal double-precision numbers to one realized in fixed-point numbers, this
example uses the following approach:
34-50
Propose Fraction Lengths
The tool stores the results in the NoOverride run, denoted by the
NoOverride label in the Run column. The Fixed-Point tool highlights
the Up Cast block to indicate that there is an issue with this result. The
Saturations column for this result shows that the block saturated 23
times, which indicates a poor guess for its scaling.
Tip In the main toolbar, use the Show option to view only blocks that
have Overflows.
34-51
34 Automatic Data Typing
4 On the Contents pane of the Fixed-Point Tool, select the Transfer Fcn
block named Analog Plant and then click the Inspect Signal button .
The Fixed-Point Tool plots the signal associated with the plant output.
34-52
Propose Fraction Lengths
The preceding plot of the plant output signal reflects the initial guess at
scaling. The Bode plot design sought to produce a well-behaved linear
response for the closed-loop system. Clearly, the response is nonlinear.
Significant quantization effects cause the nonlinear features. An important
part of fixed-point design is finding a scaling that reduces quantization effects
to acceptable levels.
Tip Use the Fixed-Point Tool plotting tools to plot simulation results
associated with logged signal data. To view a list of all logged signals, in the
main toolbar, use the Show option and select Signal logging results.
34-53
34 Automatic Data Typing
3 On the Contents pane of the Fixed-Point Tool, select the Transfer Fcn
block named Analog Plant in the NoOverride run, and then click the
Compare Signals button .
Tip From the Simulation Data Inspector menu bar, use the zoom tools to
zoom in on an area.
34-54
Propose Fraction Lengths
34-55
34 Automatic Data Typing
Note When the maximum and minimum simulation values cover the
full, intended operating range of your design, the Fixed-Point Tool yields
meaningful automatic data typing results.
Perform automatic data typing for the Controller block. This block is a
subsystem that represents software running on the target, and it requires
optimization.
The Fixed-Point Tool analyzes the scaling of all fixed-point blocks whose:
34-56
Propose Fraction Lengths
The Fixed-Point Tool uses the minimum and maximum values stored in
the DoubleOverride run to propose each block’s data types such that
the precision is maximized while the full range of simulation values is
spanned. The tool displays the proposed data types on its Contents
pane. Now, it displays the Automatic Data Typing with Simulation
Min/Max View to provide information, such as ProposedDT,
ProposedMin, ProposedMax, which are relevant at this stage of the
fixed-point conversion.
Tip In the main toolbar, use the Show option to view the groups that
must share data types. For more information, see fxptdlg in the
Simulink Reference.
3 Review the scaling that the Fixed-Point Tool proposes. You can choose to
accept the scaling proposal for each block. On the Contents pane, select the
corresponding Accept check box. By default, the Fixed-Point Tool accepts
all scaling proposals that differ from the current scaling. For this example,
ensure that the Accept check box associated with the DoubleOverride run
is selected for each of the Controller subsystem’s blocks.
The Fixed-Point Tool applies the scaling proposals that you accepted in the
previous step to the Controller subsystem’s blocks.
34-57
34 Automatic Data Typing
The Fixed-Point Tool plots the fixed-point and double override versions
of the plant output signal, as well as their difference.
34-58
Propose Fraction Lengths
34-59
34 Automatic Data Typing
Tip Optionally, you can zoom in to view the steady-state region with
greater detail. From the Tools menu of the figure window, select Zoom
In and then drag the pointer to draw a box around the area that you want
to view more closely.
34-60
Propose Word Lengths
• Design range information and range information that the Fixed-Point Tool
or Fixed-Point Advisor collects. This collected range information can be
either simulation or derived range data.
• The signedness and fraction lengths of data types that you specify for
blocks, signal objects.
• The signedness and fraction lengths of the default data types that you
specify in the Fixed-Point Tool or Fixed-Point Advisor.
• The production hardware implementation settings specified in the
Configuration Parameters dialog box.
34-61
34 Automatic Data Typing
If the signal has no specified design information, but does have a simulation
range of [-8,8] and a derived range of [-2,2], the proposal uses the union
of the ranges, [-8,8]. If you specify a safety margin of 50%, the proposal
uses a range of [-12, 12].
34-62
Propose Word Lengths
34-63
34 Automatic Data Typing
4 If you have safety margins to apply, set Safety margin for design and
derived min/max (%) and Safety margin for design and derived
min/max (%), as applicable.
Note When the Fixed-Point Tool proposes data types, it does not alter
your model.
If there are conflicts in your model, the Fixed-Point Tool opens the Result
Details dialog box.
If you do not see this warning, there are no conflicts in your model. Review
the proposed word lengths,
34-64
Propose Word Lengths
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_moving_average
Some blocks in the model already have specified fixed-point data types.
Gain3 fixdt(1,32,17)
34-65
34 Automatic Data Typing
Add1 fixdt(1,32,17)
Add2 fixdt(1,32,17)
Add3 fixdt(1,32,17)
34-66
Propose Word Lengths
The Fixed-Point Tool simulates the model and displays the results on the
Contents pane in the run named DoubleOverride.
The Fixed-Point Tool uses available range data to calculate data type
proposals according to the following rules:
• Design minimum and maximum values take precedence over the
simulation range.
The Safety margin for design and derived min/max (%) parameter
specifies a range that differs from that defined by the design range. In
this example, no safety margins are set.
• The tool observes the simulation range because you selected the
Simulation min/max option.
The Safety margin for simulation min/max (%) parameter specifies
a range that differs from that defined by the simulation range. In this
example, no safety margins are set.
34-67
34 Automatic Data Typing
The Fixed-Point Tool analyzes the data types of all fixed-point blocks whose:
• Lock output data type setting against changes by the fixed-point
tools parameter is not selected.
• Output data type parameter specifies a generalized fixed-point
number.
• Data types are not inherited types.
For each object in the model, the Fixed-Point Tool proposes the minimum
word length that avoids overflow for the collected range information.
Because the target hardware is a 16–bit embedded processor, the
Fixed-Point tool proposes word lengths based on the number of bits used
by the processor for each data type. For more information, see “How the
Fixed-Point Tool Uses Target Hardware Information” on page 34-62.
The tool proposes smaller word lengths for Gain4 and Gain4:Gain. The
tool calculated that their ideal word length is less than or equal to the
character bit length for the embedded processor (8), so the tool rounds up
the word length to 8.
34-68
Propose Word Lengths
8 To see how the target hardware affects the word length proposal, change
the target hardware to FPGA/ASIC.
a In the model, select Simulation > Model Configuration Parameters.
b In the Configuration Parameters dialog box, select Hardware
Implementation.
c On the Hardware Implementation pane, set Device vendor to
ASIC/FPGA. Simulink automatically sets the Device type to ASIC/FPGA.
d Click Apply and close the Configuration Parameters dialog box.
34-69
34 Automatic Data Typing
34-70
Propose Data Types Using Multiple Simulations
34-71
34 Automatic Data Typing
34-72
Propose Data Types Using Multiple Simulations
1 “Simulate the Model Using Random Uniform Noise” on page 34-74. Using
the Fixed-Point Tool, you simulate the model with the random uniform
noise signal and observe the simulation minimum and maximum values for
the Add block. The Fixed-Point Tool uses these simulation settings:
• Fixed-point instrumentation mode: Minimums, maximums and
overflows
• Data type override: Double
• Data type override applies to: All numeric types
• Merge instrumentation results from multiple simulations is not
selected.
This run provides the simulation results for the random uniform noise
input only.
This run provides the simulation range for the band-limited white noise
input only.
34-73
34 Automatic Data Typing
This run provides the simulation range based on the entire set of input
data for both noise sources.
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_fpt_merge
2 From the model main menu, select Analysis > Fixed-Point Tool.
34-74
Propose Data Types Using Multiple Simulations
data. The tool stores the results in a run named DoubleOverride, denoted
by the DoubleOverride label in the Run column.
5 The SimMin and SimMax values for the Add block are:
SimMin is -3.5822
SimMax is 2.7598
3 The changed values for SimMin and SimMax for the Add block are:
SimMin is now-2.5317
Merge Results
1 In the model, double-click the switch to select the random uniform noise
signal.
3 The SimMin and SimMax values for the Add block now cover the entire
simulation range for both the random uniform and band-limited white
noise signals.
SimMin is -3.5822
SimMax is 3.1542
34-75
34 Automatic Data Typing
1 On the Automatic data typing for selected system pane, click the
Propose fraction lengths button.
The Fixed-Point Tool analyzes the data types of all fixed-point blocks whose:
• Lock output data type setting against changes by the fixed-point
tools parameter is not selected.
• Output data type parameter specifies a generalized fixed-point
number.
• Data types are not inherited.
The Fixed-Point Tool uses the merged minimum and maximum values to
propose fraction lengths for each block. These values ensure maximum
precision while spanning the full range of simulation values. The tool
displays the proposed data types in the Contents pane.
34-76
View Simulation Results
Compare Runs
To compare runs:
1 In one of the runs that you want to compare, select a logged signal.
2 From the Fixed-Point Tool menu, select Results > Compare Runs or
click .
3 If there are more than two runs, in the Compare Runs Selector dialog
box, select the run that you want to compare, and then click OK.
On the upper axes, the Simulation Data Inspector plots the signal in both
selected runs. On the lower axes, the Simulation Data Inspector plots the
difference between those runs.
34-77
34 Automatic Data Typing
Compare Signals
To compare signals:
1 In one of the runs that you want to compare, select a logged signal.
2 From the Fixed-Point Tool menu, select Results > Compare Signals
or click .
3 If there are more than two runs, in the Compare Runs Selector dialog
box, select the run that you want to compare, and then click OK.
34-78
View Simulation Results
On the upper axes, the Simulation Data Inspector plots the signal in both
selected runs. On the lower axes, the Simulation Data Inspector plots the
difference between those runs.
34-79
34 Automatic Data Typing
Inspect Signals
To inspect a signal:
2 From the Fixed-Point Tool menu, select Results > Inspect Signal or click
34-80
View Simulation Results
2 From the Fixed-Point Tool menu, select Results > Histogram Plot of
Signal or click .
The histogram plot helps you visualize the dynamic range of a signal. It
provides information about the:
You can use this information to estimate the word size required to represent
the signal.
34-81
34 Automatic Data Typing
See Also
• “Viewing Results With the Simulation Data Inspector” on page 34-83
• “Propose Fraction Lengths Using Simulation Range Data” on page 34-50
• “Converting a Model from Floating- to Fixed-Point Using Simulation Data”
on page 30-11
34-82
Viewing Results With the Simulation Data Inspector
34-83
34 Automatic Data Typing
Tip The Contents pane of the Fixed-Point Tool displays an antenna icon
next to items that you can plot.
See Also
• “Validate System Behavior”
• fxptdlg
34-84
35
Range Analysis
• Inputs
• The semantics of each calculation in the blocks
35-2
How Range Analysis Works
If the model contains objects that the analysis cannot support, where possible,
the software uses automatic stubbing. For more information, see “Automatic
Stubbing” on page 35-5.
The range analysis tries to narrow the derived range by using all the specified
design ranges in the model. The more design range information you specify,
the more likely the range analysis is to succeed. As the software performs the
analysis, it derives new range information for the model. The software then
attempts to use this new information, together with the specified ranges, to
derive ranges for the remaining objects in the model.
35-3
35 Range Analysis
The following table summarizes how the analysis derives range information
and provides links to examples.
• Specified minimum
and maximum values
• Derived minimum
and maximum values
A parameter on a block The analysis takes both “Derive Ranges
has initial conditions factors into account by Using Block Initial
and a design range. taking the union of the Conditions” on page
design range and the 35-16
initial conditions.
The model contains The analysis does not “Derive Ranges Using
a parameter with a take into account the Design Ranges for
specified range and the range specified for the Simulink.Parameter
parameter storage class parameter. Instead, Objects” on page 35-19
is set to Auto. it uses the parameter
value.
The model contains The analysis takes “Derive Ranges Using
a parameter with a into account the Design Ranges for
specified range and the range specified for Simulink.Parameter
parameter storage class the parameter and Objects” on page 35-19
is not set to Auto. ignores the value.
35-4
How Range Analysis Works
Automatic Stubbing
35-5
35 Range Analysis
The software cannot “stub” all Simulink blocks, such as the Integrator block.
See the blocks marked “not stubbable” in “Supported and Unsupported
Simulink Blocks” on page 35-46.
35-6
Derive Ranges
Derive Ranges
1 Verify that your model is compatible with range analysis. See “Model
Compatibility with Range Analysis” on page 35-6.
2 In Simulink, open your model and set it up for use with the Fixed-Point
Tool. For more information, see “Set Up the Model” on page 34-28.
6 In the Settings for selected system pane, set Data type override to
Double, then click Apply.
35-7
35 Range Analysis
35-8
Derive Ranges
8 In the Fixed-Point Tool, click the Derive min/max values for selected
system button.
The analysis runs and tries to derive range information for objects in the
selected system. Your next steps depend on the analysis results.
35-9
35 Range Analysis
For example, consider a subsystem that has an input with a design minimum
of -10 and a design maximum of 10 that is connected to an input signal with
a constant value of 1. When you derive ranges at the model level, the range
analysis software uses the constant value 1 as the input. When you derive
ranges at the subsystem level, the range analysis software does not take the
constant value into account and instead uses [-10..10] as the range.
• System validation
It is a best practice to analyze individual subsystems in your model one at a
time. This practice makes it easier to understand the atomic behavior of
the subsystem. It also makes debugging easier by isolating the source of
any issues.
• Calibration
The results from the analysis at subsystem level are based only on the
settings specified within the subsystem. The proposed data types cover
the full intended design range of the subsystem. Based on these results,
35-10
Derive Ranges at the Subsystem Level
you can determine whether you can reuse the subsystem in other parts of
your model.
To derive ranges at the subsystem level, the key points to remember are:
Tip If the parent of the selected subsystem controls the data type override
setting of the subsystem, first set the parent Data type override to Use
local settings and then set the subsystem Data type override to
Double.
35-11
35 Range Analysis
35-12
View Derived Ranges in the Fixed-Point Tool
35-13
35 Range Analysis
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_derived_min_max_1
35-14
Range Analysis Examples
The model displays the specified design minimum and maximum values for
each block.
• In1 design range is [-50..100].
• In2 design range is [-50..35].
• Add block design range is [-125..55].
35-15
35 Range Analysis
Tip To display design ranges in your model, from the model menu, select
Display > Signals & Ports and select Design Ranges.
3 In the Settings for selected system pane, set Data type override to
Double, then click Apply.
4 In the Fixed-Point Tool, click the Derive min/max values for selected
system button.
To calculate the derived range at the Add block input, the software uses
the design minimum and maximum values specified for the Inport blocks,
[-50..100] and [-50..35]. The derived range at the Add block input
is [-85..150].
In the Contents pane, the Fixed-Point Tool displays the derived and design
minimum and maximum values for the blocks in the selected system.
• The derived range for the Add block output signal is narrowed to
[-85..55]. This derived range is the intersection of the range derived
from the block inputs, [-85..150], and the design minimum and
maximum values specified for the block output, [-125..55].
• The derived range for the Outport block Out1 is [-85..55], the same as
the Add block output.
35-16
Range Analysis Examples
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_derived_min_max_2
The model uses block annotations to display the specified design minimum
and maximum values for the Inport block and the initial conditions for
the Unit Delay block.
35-17
35 Range Analysis
Tip To display design ranges in your model, from the model menu, select
Display > Signals & Ports and select Design Ranges.
3 In the Settings for selected system pane, set Data type override to
Double, then click Apply.
4 In the Fixed-Point Tool, click the Derive min/max values for selected
system button.
In the Contents pane, the Fixed-Point Tool displays the derived minimum
and maximum values for the blocks in the model.
The derived minimum and maximum range for the Outport block, Out1,
is [0..10]. The range analysis derives this range by taking the union of
the initial value, 0, on the Unit Delay block and the design range on the
block, [5..10].
Because the analysis takes the union of the initial conditions, 7, and the
design range, [5..10], on the Unit Delay block, the derived range for
the block is still [5..10].
35-18
Range Analysis Examples
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
35-19
35 Range Analysis
ex_derived_min_max_3
The model displays the specified design minimum and maximum values for
the Inport blocks. The design range for both Inport blocks is [1..2].
35-20
Range Analysis Examples
Tip To display design ranges in your model, from the model menu, select
Display > Signals & Ports and select Design Ranges.
a Double-click each Gain block and note the name of the Gain parameter
on the Main tab.
Gain1 paramObjOne
Gain2 paramObjTwo
4 In the Settings for selected system pane, set Data type override to
Double , then click Apply.
5 In the Fixed-Point Tool, click the Derive min/max values for selected
system button.
35-21
35 Range Analysis
In the Contents pane, the Fixed-Point Tool displays the derived minimum
and maximum values for the blocks in the model.
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_derived_min_max_4
35-22
Range Analysis Examples
The model displays the specified design minimum and maximum values
for the blocks in the model.
• The Inport block In1 has a design minimum of -1 but no specified
maximum value, as shown by the annotation, Max=[].
• The Gain block has a design range of [-1.5..1.5].
• The Outport block Out1 has no design range specified, as shown by the
annotations, Min=[], Max=[].
35-23
35 Range Analysis
Tip To display design ranges in your model, from the model menu select
Display > Signals & Ports and select Design Ranges.
3 In the Settings for selected system pane, set Data type override to
Double, then click Apply.
4 In the Fixed-Point Tool, click the Derive min/max values for selected
system button.
In the Contents pane, the Fixed-Point Tool displays the derived minimum
and maximum values for the blocks in the model. The range analysis is
unable to derive a maximum value for the Inport block, In1. The tool
highlights this result.
b In the block parameters dialog box, select the Signal Attributes tab.
The model displays the updated maximum value in the block annotation
for In1.
35-24
Range Analysis Examples
6 In the Fixed-Point Tool, click the Derive min/max values for selected
system button to rerun the range analysis.
The range analysis can now derive ranges for the Inport and Gain blocks.
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_derived_min_max_5
35-25
35 Range Analysis
The model displays the specified design minimum and maximum values
for the blocks in the model.
• The Inport block In1 has a design range of -10,20.
• The rest of the blocks in the model have no specified design range.
Tip To display design ranges in your model, from the model menu select
Display > Signals & Ports and select Design Ranges.
35-26
Range Analysis Examples
3 In the Settings for selected system pane, set Data type override to
Double , then click Apply.
4 In the Fixed-Point Tool, click the Derive min/max values for selected
system button.
In the Contents pane, the Fixed-Point Tool displays the derived minimum
and maximum values for the blocks in the model. Because one of the Add
block inputs is fed back from its output, the analysis is unable to derive an
output range for the Add block or for any of the blocks connected to this
output. The Fixed-Point Tool highlights these results.
5 To fix the issue, specify design minimum and maximum values for Out1:
b In the block parameters dialog box, select the Signal Attributes tab.
c On this tab, set Minimum to -20 and Maximum to 40 and click OK.
6 In the Fixed-Point Tool, click the Derive min/max values for selected
system button to rerun the range analysis.
The range analysis uses the minimum and maximum values specified for
Out1, [-20..40] and the gain value of Gain3, 2, to derive an input range
35-27
35 Range Analysis
for Gain3, [-10..20]. Because the input of Gain3 feeds back to the input of
the Add block, the analysis now derives ranges for all objects in the model.
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_range_conflict
35-28
Range Analysis Examples
The model displays the specified design minimum and maximum values
for the blocks in the model.
• The Inport blocks In1 and In2 have a design range of [-1..1].
• The Outport block Out1 has a design range of [10..20].
Tip To display design ranges in your model, from the model menu select
Display > Signals & Ports and select Design Ranges.
35-29
35 Range Analysis
3 In the Settings for selected system pane, set Data type override to
Double , then click Apply.
4 In the Fixed-Point Tool, click the Derive min/max values for selected
system button.
The Fixed-Point Tool generates an error because the range analysis fails.
It reports an error because the derived range for the Sum block, [-2..2] is
outside the specified design range for the Outport block, [10..20].
6 To fix the conflict, change the design range on the Outport block to
[-10..20] so that this range includes the derived range for the Sum block.
a In the model, double-click the Outport block.
b In the block parameters dialog box, click the Signal Attributes tab.
7 In the Fixed-Point Tool, click the Derive min/max values for selected
system button to rerun the range analysis.
35-30
Derive Ranges for a Referenced Model
Derive Ranges
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_derived_sum_multi_instance
35-31
35 Range Analysis
The model displays the specified design minimum and maximum values
for the blocks in the model.
• The Inport block In1 has a design range of [-50..100].
• The Inport block In2 has a design range of [-50..35].
Tip To display design ranges in your model, from the model menu select
Display > Signals & Ports and select Design Ranges.
The model contains two Model blocks that both reference the ex_sum model.
35-32
Derive Ranges for a Referenced Model
35-33
35 Range Analysis
4 In the Settings for selected system pane, set Data type override to
Double.
5 In the Fixed-Point Tool Data collection pane, set Store results in run to
double_run and then click Apply.
Providing a unique name for the run avoids accidentally overwriting results
from previous runs and enables you to identify the run more easily.
6 In the Fixed-Point Tool Model Hierarchy pane, select the ex_sum model.
The Data type override setting is Off. The setting in the parent model
does not affect the setting in the referenced model — you must change it
manually in the referenced model.
7 For the ex_sum model, set Data type override to Double and then click
Apply.
Changing the setting for any instance of the referenced model changes the
setting on all instances and on the referenced model itself.
The Fixed-Point Tool cannot derive ranges if your model contains unsaved
changes.
To calculate the derived ranges, the software uses the design minimum and
maximum values specified for the Inport blocks in the top-level model,
In1 and In2.
In the Contents pane, the Fixed-Point Tool displays the derived and design
minimum and maximum values for the blocks and referenced models in
the ex_derived_sum_multi_instance model. Some of the derived values
that the Fixed-Point Tool reports are slightly larger than expected. This
difference is due to rounding errors because the software approximates
35-34
Derive Ranges for a Referenced Model
1 In the Model Hierarchy pane, select the first instance of the referenced
model, Model(ex_sum).
The tool displays the derived minimum and maximum values for this
instance of the referenced model, [-82.001..140.001]. This range
is derived from the outputs of the two Gain blocks, [-40,80] and
[-60..42.001].
The tool displays the derived values for the second instance, [-85..150].
This range is derived from the referenced model inputs, In1 and In2,
[-50..100] and [-50..35] respectively.
For this node, the Fixed-Point Tool displays the merged results for the
derived range for the referenced model which is the union of the results for
each instance of the model, [-85..150].
Next, you set design range on Sum block in referenced model to see how the
range analysis takes this information into account.
35-35
35 Range Analysis
2 In the block parameters dialog box, on the Signal Attributes tab, set
Output minimum to -125 and Output maximum to 50, click OK, and
then save the model.
This time, to calculate the derived ranges, the software uses the design
minimum and maximum values specified for the Inport blocks in the
parent model, In1 and In2, and the design minimum and maximum values
specified for the Sum block in the referenced model.
4 You can now propose data types for the model based on these derived
minimum and maximum values.
35-36
Propose Data Types for a Referenced Model
2 In the Automatic data typing for selected system pane, click the
Configure link, set Default data type of all floating-point signals to
fixdt(1,16,4) and then click Apply.
The Fixed-Point Tool proposes fraction lengths for the inputs In1 and In2
based on the design minimum and maximum values specified on the blocks
in the model and on the derived minimum and maximum values.
The tool does not propose data types for the other blocks because they use
inherited data types. Instead, it displays n/a in the ProposedDT column.
The Fixed-Point Tool might not be able to propose data types for other
reasons, to view more information, click the Show details for selected
result button .
The tool displays the proposed scaling in its Contents pane. It displays the
Automatic Data Typing View to provide information, such as ProposedDT,
ProposedMin, ProposedMax, which are relevant at this stage of the
fixed-point conversion.
35-37
35 Range Analysis
4 After reviewing the data type proposals, click Apply accepted fraction
lengths to apply the proposed data types to your model.
35-38
Deriving Ranges for a Referenced Model
The Fixed-Point Tool displays the top-level model that contains the referenced
models and the referenced models in its Model Hierarchy pane. For
example, the ex_derived_sum_multi_instance model contains two instances
of the referenced model ex_sum. The Fixed-Point Tool displays both models
and both instances of the referenced model in the model hierarchy.
35-39
35 Range Analysis
In the referenced model node, the tool displays the union of the results for
each instance of the referenced model.
When you change the fixed-point instrumentation and data type override
settings for any instance of a referenced model, the settings change on all
instances of the model and on the referenced model itself.
See Also
• “Derive Ranges for a Referenced Model” on page 35-31
35-40
Intermediate Range Results
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_intermediateRange
35-41
35 Range Analysis
1 Press CTRL+D to update the diagram. Notice the design range information
for each of the input ports.
The Fixed-Point Tool Contents pane displays the derived minimum and
maximum values for each object in the ex_intermediateRange model.
The proposed data types appear in the Contents pane of the Fixed-Point
Tool.
6 Look at the proposed data type of the Product block. The Fixed-Point Tool
proposed a data type with 32–bit word length and 11–bit fraction length.
The derived maximum value of the Product block is 1, but the maximum
representable value for the proposed data type is approximately 1,048,575.
To learn more about the data type proposal, in the Contents pane,
right-click on the product block and select Result Details.
7 In the Result Details dialog box, notice the rows labeled Intermediate
Maximum and Intermediate Minimum. After the first two inputs to the
Product block are multiplied, the block has a maximum value of 1000000
before being multiplied by the next two inputs for a final maximum value
of 1. The data type proposal for the Product block in this model is based
on the intermediate minimum and maximum values. It is not based on
the derived minimum and maximum values to prevent overflows at the
intermediate stages of the block.
35-42
Intermediate Range Results
35-43
35 Range Analysis
35-44
Unsupported Simulink® Software Features
Signals with nonzero The software does not support models with signals
sample time offset that have nonzero sample time offsets.
Models with no The software only supports models that have one or
output ports more output ports.
35-45
35 Range Analysis
The software supports all blocks in the Additional Math and Discrete library.
The Commonly Used Blocks library includes blocks from other libraries.
Those blocks are listed under their respective libraries.
Continuous Library
35-46
Supported and Unsupported Simulink® Blocks
Discontinuities Library
Discrete Library
Memory Supported
Tapped Delay Supported
35-47
35 Range Analysis
The software supports all blocks in the Logic and Bit Operations library.
or
35-48
Supported and Unsupported Simulink® Blocks
or
35-49
35 Range Analysis
35-50
Supported and Unsupported Simulink® Blocks
35-51
35 Range Analysis
35-52
Supported and Unsupported Simulink® Blocks
35-53
35 Range Analysis
35-54
Supported and Unsupported Simulink® Blocks
Sinks Library
Sources Library
35-55
35 Range Analysis
function y = fcn(u,v)
%#codegen
y = 2*u;
y = y + v;
35-56
Supported and Unsupported Simulink® Blocks
35-57
35 Range Analysis
• The parent model and the referenced model have mismatched data type
override settings. The data type override setting of the parent model and
all of its referenced models must be the same, unless the data type override
setting of the parent model is Use local settings. You can select the
data type override settings for your model in the Analysis menu, in the
Fixed Point Tool dialog box under the Settings for selected system pane.
35-58
36
• “Refine Data Types of a Model with Buses Using Simulation Data” on page
36-2
• “Convert a Model with Buses to Fixed-Point Using Range Analysis” on
page 36-13
• “Bus Objects in the Fixed-Point Workflow” on page 36-24
• “” on page 36-1
36 Working with Bus Objects in the Fixed-Point Workflow
This example shows how to refine the fixed-point types of a model with bus
objects based on simulation data.
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_bus_Fix
1 Press ctrl+d to update the diagram. Notice the use of fixed-point data
types in the model.
36-2
Refine Data Types of a Model with Buses Using Simulation Data
36-3
36 Working with Bus Objects in the Fixed-Point Workflow
36-4
Refine Data Types of a Model with Buses Using Simulation Data
2 Obtain the ideal numerical behavior of the model by overriding all of the
data types with floating-point singles.
In the Settings for selected system pane, set Data type override to
Single. Set Data type override applies to to All numeric types.
The model is simulated using floating-point singles, so this run does not
contain any overflows.
4 Return to the model. A prefix has been added to the names on the bus
signals. This prefix indicates that the bus objects are replaced by temporary
overridden bus objects specifying single data types. For more information,
see “Bus Naming Conventions with Data Type Override” on page 36-24.
Overridden bus objects are not stored in the base workspace, so you cannot
see them in the Bus Editor. The Bus Editor continues to display the data
types of the bus objects before they were overridden.
36-5
36 Working with Bus Objects in the Fixed-Point Workflow
2 In the Propose Data Types dialog box, select SingleOverride as the run
to use for proposing data types. Click OK.
The proposed data types appear in the Contents pane of the Fixed-Point
Tool.
36-6
Refine Data Types of a Model with Buses Using Simulation Data
1 On the Fixed-Point Tool toolbar, select Show > Groups that must share
the same data type to better understand the proposed data types.
2 In the Contents pane, right-click the column labeled DTGroup and select
Group by This Column.
The Fixed-Point Tool organizes the Contents pane to group the elements
by data type.
In the Result Details dialog box, view the details of the proposed data
type for the bus element. The data type proposal for a bus element is found
by taking the union of the ranges of all blocks that must share the same
data type, which includes all of the sources driving the bus element. The
software then proposes a data type for this range.
36-7
36 Working with Bus Objects in the Fixed-Point Workflow
36-8
Refine Data Types of a Model with Buses Using Simulation Data
36-9
36 Working with Bus Objects in the Fixed-Point Workflow
3 In the Fixed-Point Tool Contents pane, verify that there are no overflows
in Fixed_Run.
36-10
Refine Data Types of a Model with Buses Using Simulation Data
5 Return to the model. Notice that the bus objects no longer contain the
prefixes.
36-11
36 Working with Bus Objects in the Fixed-Point Workflow
Open the Bus Editor by clicking Edit > Bus Editor. You can now view the
new, refined data types of the bus objects in the Bus Editor.
36-12
Convert a Model with Buses to Fixed-Point Using Range Analysis
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_bus_Flt
1 Press ctrl+d to update the diagram and view the data types of each block
in the model.
36-13
36 Working with Bus Objects in the Fixed-Point Workflow
2 Under the Signal Attributes tab, notice that the design minimum and
maximum have been set to -1 and 1, respectively.
36-14
Convert a Model with Buses to Fixed-Point Using Range Analysis
36-15
36 Working with Bus Objects in the Fixed-Point Workflow
The Address unsupported blocks check warns that the Sine blocks do
not support fixed-point data types. Because the Sine blocks are decoupled
from the model with Data Type Conversion blocks, you do not need to take
any further action.
The Check bus usage check advises you to configure your model to detect
future changes that can result in improper bus usage.
Click OK and run the check again from the Fixed-Point Advisor.
This time the check passes. All other checks under Verify Fixed-Point
Conversion Guidelines pass.
5 Run each check in the Prepare for Data Typing and Scaling folder.
The Verify hardware selection check fails. Because the target hardware
is an embedded processor, the Fixed-Point Advisor recommends that you
set this value to the hardware integer used by the embedded hardware. Set
Default data type of all floating point signals to Same as embedded
hardware integer.
36-16
Convert a Model with Buses to Fixed-Point Using Range Analysis
All other checks under Prepare for Data Typing and Scaling pass.
The Fixed-Point Tool Contents pane displays the derived minimum and
maximum values for each object in the ex_bus_Flt model.
The proposed data types appear in the Contents pane of the Fixed-Point
Tool.
1 On The Fixed-Point Tool toolbar, select Show > Groups that must share
the same data type to better understand the proposed data types.
2 In the Contents pane, right-click the column labeled DTGroup and select
Group by This Column.
The Fixed-Point Tool organizes the contents pane to group the elements
by data type.
36-17
36 Working with Bus Objects in the Fixed-Point Workflow
In the Result Details dialog box, view the details of the proposed data
type for the bus element. The software finds the data type proposal for
a bus element by taking the union of the ranges of all blocks that must
share the same data type, which includes all of the sources driving the bus
element. It then proposes a data type for this range.
36-18
Convert a Model with Buses to Fixed-Point Using Range Analysis
36-19
36 Working with Bus Objects in the Fixed-Point Workflow
36-20
Convert a Model with Buses to Fixed-Point Using Range Analysis
2 In the Fixed-Point Tool Contents pane, verify that there are no overflows
in Fixed_Run.
36-21
36 Working with Bus Objects in the Fixed-Point Workflow
36-22
Convert a Model with Buses to Fixed-Point Using Range Analysis
4 Open the Bus Editor from the model menu by clicking Edit > Bus Editor.
You can now view the new fixed-point data types of the bus elements in
the Bus Editor.
36-23
36 Working with Bus Objects in the Fixed-Point Workflow
DTO Applies To
All numeric
DTO Mode types Floating point Fixed point
Scaled Double dtoScl_myBus dtoSclFlt_myBus dtoSclFxp_myBus
Double dtoDbl_myBus dtoDblFlt_myBus dtoDblFxp_myBus
Single dtoSgl_myBus dtoSglFlt_myBus dtoSglFxp_myBus
Note You cannot see bus objects with an overridden data type within the Bus
Editor because they are not stored in the base workspace.
• A model in accelerator mode that has a bus object with an overridden data
type at the output port.
To perform data type override, run the model in normal mode.
36-24
Bus Objects in the Fixed-Point Workflow
The following example shows how the software determines the data types
for elements of bus objects.
36-25
36 Working with Bus Objects in the Fixed-Point Workflow
Each of the four input ports in this model have specified design ranges. The
In2 and In4 input ports must share the same data type because they drive the
same element of the mybus bus object.
The Fixed-Point Tool proposes a data type based on the union of these two
ranges. After proposing data types for the model, in the Result Details
dialog box for the mybus : b element of the bus object, notice the field
labeled Shared Design Maximum and Shared Design Minimum. The
proposed data type is based on this range, which is the union of the design
ranges of the In2 and In4 blocks.
36-26
Bus Objects in the Fixed-Point Workflow
36-27
36 Working with Bus Objects in the Fixed-Point Workflow
36-28
37
Code Generation
You can generate code for testing on a rapid prototyping system using products
such as Simulink Real-Time™, Real-Time Windows Target™, or dSPACE®
software. The target compiler and processor may support floating-point
operations in software or in hardware. In any case, the fixed-point portions of
a model generate pure integer code and do not use floating-point operations.
This allows valid bit-true testing even on a floating-point processor.
You can also generate code for non-real-time testing. For example, you can
generate code to run in nonreal time on computers running any supported
operating system. Even though the processors have floating-point hardware,
the code generated by fixed-point blocks is pure integer code. The Generic
Real-Time Target (GRT) in the Simulink Coder product and acceleration
modes in the Simulink software are examples of where non-real-time code
is generated and run.
When used with HDL Coder, Fixed-Point Designer lets you generate bit-true
synthesizable Verilog® and VHDL® code from Simulink models, Stateflow
charts, and MATLAB Function blocks.
37-2
Code Generation Support
Introduction
All fixed-point blocks support code generation, except particular simulation
features. The sections that follow describe the code generation support that
the Fixed-Point Designer software provides. You must have a Simulink Coder
license to generate C code or a HDL Coder license to generate HDL code.
Languages
C code generation is supported.
Data Types
Fixed-point code generation supports all integer and fixed-point data types
that are supported by simulation. See “Supported Data Types” on page 26-20.
Rounding Modes
All rounding modes—Ceiling, Convergent, Floor, Nearest, Round,
Simplest, and Zero —are supported.
Overflow Handling
• Saturation and wrapping are supported.
37-3
37 Code Generation
Blocks
All blocks generate code for all operations with a few exceptions. The
Lookup Table Dynamic block generates code for all lookup methods except
Interpolation-Extrapolation.
Scaling
Any binary-point-only scaling and [Slope Bias] scaling that is supported in
simulation is supported, bit-true, in code generation.
37-4
Accelerating Fixed-Point Models
Note Rapid Accelerator mode does not support models with bus objects or
33+ bit fixed-point data types as parameters.
Every Simulink model is configured to have a start time and a stop time in
the Configuration Parameters dialog box. Simulink simulations are usually
configured for non-real-time execution, which means that the Simulink
software tries to simulate the behavior from the specified start time to the
stop time as quickly as possible. The time it takes to complete a simulation
consists of two parts: overhead time and core simulation time, which is spent
calculating changes from one time step to the next. For any model, the time
it takes to simulate if the stop time is the same as the start time can be
regarded as the overhead time. If the stop time is increased, the simulation
takes longer. This additional time represents the core simulation time. Using
an acceleration mode to simulate a model has an initially larger overhead
time that is spent generating and compiling code. For any model, if the
simulation stop time is sufficiently close to the start time, then Normal mode
simulation is faster than an acceleration mode. But an acceleration mode
can eliminate the overhead of code generation for subsequent simulations if
structural changes to the model have not occurred.
In Normal mode, the Simulink software runs general code that can handle
various situations. In an acceleration mode, code is generated that is tailored
to the current usage. For fixed-point use, the tailored code is much leaner
than the simulation code and executes much faster. The tailored code allows
an acceleration mode to be much faster in the core simulation time. For any
model, when the stop time is close to the start time, overhead dominates the
overall simulation time. As the stop time is increased, there is a point at
which the core simulation time dominates overall simulation time. Normal
37-5
37 Code Generation
Normal mode generally uses more tailored code for floating-point calculations
compared to fixed-point calculations. Normal mode is therefore generally
much faster for floating-point models than for similar fixed-point models.
For acceleration modes, the situation often reverses and fixed point becomes
significantly faster than floating point. As noted above, the fixed-point code
goes from being general to highly tailored and efficient. Depending on the
hardware, the integer-based fixed-point code can gain speed advantages over
similar floating-point code. Many processors can do integer calculations much
faster than similar floating-point operations. In addition, if the data bus is
narrow, there can also be speed advantages to moving around 1-, 2-, or 4-byte
integer signals compared to 4- or 8-byte floating-point signals.
37-6
Using External Mode or Rapid Simulation Target
Introduction
If you are using the Simulink Coder external mode or rapid simulation (rsim)
target, there are situations where you might get unexpected errors when
tuning block parameters. These errors can arise when you specify the Best
precision scaling option for blocks that support constant scaling for best
precision. See “Constant Scaling for Best Precision” on page 27-15 for a
description of the constant scaling feature.
The sections that follow provide further details about the errors you might
encounter. To avoid these errors, specify a scaling value instead of using
the Best precision scaling option.
External Mode
If you change a parameter such that the binary point moves during an
external mode simulation or during graphical editing, and you reconnect to
the target, a checksum error occurs and you must rebuild the code. When you
use Best Precision scaling, the binary point is automatically placed based
on the value of a parameter. Each power of two roughly marks the boundary
where a parameter value maps to a different binary point. For example, a
parameter value of 1 to 2 maps to a particular binary point position. If you
change the parameter to a value of 2 to 4, the binary point moves one place to
the right, while if you change the parameter to a value of 0.5 to 1, it moves
one place to the left.
For example, suppose a block has a parameter value of -2. You then build
the code and connect in external mode. While connected, you change the
parameter to -4. If the simulation is stopped and then restarted, this
parameter change causes a binary point change. In external mode, the binary
point is kept fixed. If you keep the parameter value of -4 and disconnect
37-7
37 Code Generation
from the target, then when you reconnect, a checksum error occurs and you
must rebuild the code.
37-8
Optimize Your Generated Code
37-9
37 Code Generation
Reduces
Model
Tip Reduces ROM
Execution
Time
“Limit the Use of Custom Storage Classes” on page 37-13 Yes No
“Limit the Use of Unevenly Spaced Lookup Tables” on page
Yes Yes
37-13
“Minimize the Variety of Similar Fixed-Point Utility
Yes No
Functions” on page 37-13
Dependent
Dependent
on model
on model
configuration,
“Handle Net Slope Correction” on page 37-14 configuration,
compiler,
compiler, and
and target
target hardware
hardware
“Optimize Generated Code Using Specified Minimum and
Yes Yes
Maximum Values” on page 37-25
For temporary variables that only occupy a CPU register or stack location
briefly, the space consumed by a long is less critical. However, depending on
the operation, the use of long variables in math operations can be expensive.
Addition and subtraction of long integers generally requires the same effort
as adding and subtracting regular integers, so that operation is not a concern.
37-10
Optimize Your Generated Code
The primary exception to this rule is the rounding behavior of signed integer
division. The C language leaves this rounding behavior unspecified, but for
most targets the “no effort” mode is round to zero. For unsigned division,
everything is nonnegative, so rounding to floor and rounding to zero are
identical.
You can improve code efficiency by setting the value of the Model
Configuration Parameters > Hardware Implementation > Production
hardware> Signed integer division rounds to parameter to describe how
your production target handles rounding for signed division. For Product
blocks that are doing only division, setting the Integer rounding mode
parameter to the rounding mode of your production target gives the best
37-11
37 Code Generation
results. You can also use the Simplest rounding mode on blocks where it
is available. For more information, refer to “Rounding Mode: Simplest” on
page 28-14.
The options for overflow handling also have a big impact on the efficiency
of your generated code. Using software to detect overflow situations and
saturate the results requires the code to be much bigger and slower compared
to simply ignoring the overflows. When overflows are ignored for unsigned
integers and two’s complement signed integers, the results usually wrap
around modulo 2N, where N is the number of bits. Unhandled overflows that
wrap around are highly undesirable for many situations.
However, because of code size and speed needs, traditional handwritten code
contains very little software saturation. Typically, the fixed-point scaling is
very carefully set so that overflow does not occur in most calculations. The
code for these calculations safely ignores overflow. To get results comparable
to or better than what you would expect from traditional handwritten code,
the Saturate on integer overflow parameter should not be selected for
Simulink blocks doing those calculations.
In a design, there might be a few places where overflow can occur and
saturation protection is needed. Traditional handwritten code includes
software saturation for these few places where it is needed. To get comparable
generated code, the Saturate on integer overflow parameter should only
be selected for the few Simulink blocks that correspond to these at-risk
calculations.
A secondary benefit of using the most efficient options for overflow handling
and rounding is that calculations often reduce from multiple statements
requiring several lines of C code to small expressions that can be folded into
downstream calculations. Expression folding is a code optimization technique
that produces benefits such as minimizing the need to store intermediate
computations in temporary buffers or variables. This can reduce stack size
and make it more likely that calculations can be efficiently handled using only
CPU registers. An automatic code generator can carefully apply expression
folding across parts of a model and often see optimizations that might not be
obvious. Automatic optimizations of this type often allow generated code to
exceed the efficiency of typical examples of handwritten code.
37-12
Optimize Your Generated Code
You can access the Signal Properties dialog box by selecting any connection
between blocks in your model, and then selecting Signal Properties from
the Simulink Edit menu.
37-13
37 Code Generation
The best way to identify these opportunities is to inspect the generated code.
For each utility function that appears in the generated code, you can search
for all the call sites. If relatively few calls to the function are made, then trace
back from the call site to the Simulink model. By modifying those places in
the Simulink model, it is possible for you to eliminate the few cases that
need a rarely used utility function.
When a change of fixed-point slope is not a power of two, net slope correction
is necessary. Normally, net slope correction is implemented using an integer
multiplication followed by shifts. Under some conditions, an alternate
implementation requires just an integer division by a constant. One of the
conditions is that the net slope can be accurately represented as the reciprocal
of an integer. Under this condition, the division implementation gives more
accurate numerical behavior. Depending on your compiler and embedded
hardware, the division implementation might be more desirable than the
multiplication and shifts implementation. The generated code for the division
implementation might require less ROM or improve model execution time.
37-14
Optimize Your Generated Code
• The software cannot perform the division using the production target long
data type and therefore must use multiword operations.
Using multiword division does not produce code suitable for embedded
targets. Therefore, do not use integer division to handle net slope correction
in models that use multiword operations. If your model contains blocks
that use multiword operations, change the word length of these blocks
to avoid these operations.
• Net slope is a power of 2
Binary-point-only scaling, where the net slope is a power of 2, involves
moving the binary point within the fixed-point word. This scaling mode
already minimizes the number of processor arithmetic operations.
1 Select the Optimization > Use integer division to handle net slopes
that are reciprocals of integers configuration parameter.
For more information, see “Use integer division to handle net slopes that
are reciprocals of integers”.
37-15
37 Code Generation
Note You must set this parameter to a value that is appropriate for the
target hardware. Failure to do so might result in division operations that
comply with the definition on the Hardware Implementation pane, but
are inappropriate for the target hardware.
3 Set the Integer rounding mode of the blocks that require net slope
correction (for example, Product, Gain, and Data Type Conversion) to
Simplest or match the rounding mode of your target hardware.
Note You can use the Model Advisor to alert you if you have not configured
your model correctly for this optimization. Open the Model Advisor and
run the Identify questionable fixed-point operations check. For more
information, see “Use the Model Advisor to Optimize Fixed-Point Operations
in Generated Code” on page 37-41.
37-16
Optimize Your Generated Code
These values are represented by the general [Slope Bias] encoding scheme
described in “Scaling” on page 27-6: Vi = Si Qi + Bi .
Sa Qa = SbQb .Sc Qc , or
37-17
37 Code Generation
Sb Sc
Qa = .QbQc
Sa
where the net slope is:
Sb Sc
Sa
The net slope for the Product block is 1/1000. Because the net slope is the
reciprocal of an integer, you can use the Use integer division to handle
net slopes that are reciprocals of integers optimization parameter if your
model and hardware configuration are suitable. For more information, see
“When to Use Integer Division to Handle Net Slope Correction” on page 37-14.
1 For the two Constant blocks, set the Output data type to fixdt(1, 16,
1/1000, 0).
2 For the Product block, set the Output data type to fixdt(1, 16,
1/1000, 0). Set the Integer rounding mode to Simplest.
4 Clear the Optimization > Use integer division to handle net slopes
that are reciprocals of integers configuration parameter.
37-18
Optimize Your Generated Code
Note You can set up the Fixed-Point Designer software to provide alerts
when precision loss occurs in fixed-point constants. For more information,
see “Net Slope and Net Bias Precision” on page 28-21.
37-19
37 Code Generation
6 Select the Optimization > Use integer division to handle net slopes
that are reciprocals of integers configuration parameter, save your
model, and simulate again.
The software implements the net slope correction using division instead of
multiplication followed by shifts. The calculated product is 4, as you expect.
37-20
Optimize Your Generated Code
Note This setting must match your target hardware rounding mode.
• The Integer rounding mode of the Product block in the model is set
to Simplest.
• The model does not use multiword operations.
Note The generated code is more efficient only if division is more efficient
than multiplication followed by shifts on your target hardware.
37-21
37 Code Generation
These values are represented by the general [Slope Bias] encoding scheme
described in “Scaling” on page 27-6: Vi = Si Qi + Bi .
Sa Qa = SbQb .Sc Qc , or
Sb Sc
Qa = .QbQc
Sa
where the net slope is:
Sb Sc
Sa
The net slope for the Product block is 1/1000.
37-22
Optimize Your Generated Code
Sa Qa + Ba = SbQb + Bb
Sb
There is no bias. Therefore, the net slope is Sa . The net slope for this block
is also 1/1000.
Because the net slope is the reciprocal of an integer, you can use the Use
integer division to handle net slopes that are reciprocals of integers
optimization parameter if your model and hardware configuration are
suitable. For more information, see “When to Use Integer Division to Handle
Net Slope Correction” on page 37-14.
1 For the two Inport blocks, U and V, set the Data type to int16.
2 For the Data Type Conversion block, set the Integer rounding mode to
Simplest. Set the Output data type to fixdt(1, 16, 1000, 0).
3 For the Product block, set the Integer rounding mode to Simplest. Set
the Output data type to fixdt(1, 16, 1000, 0).
5 Clear the Optimization > Use integer division to handle net slopes
that are reciprocals of integers configuration parameter.
6 From the Simulink model menu, select Code > C/C++ Code > Build
Model.
Yc = 0.001 * U;
Ym = 0.001 * U * V;
37-23
37 Code Generation
The ideal value of the net slope correction is 0.001. In the generated
code, the approximate value of the net slope correction is 16777L >> 24
= 16777/2^24 = 0.000999987125396729. This approximation introduces
numerical inaccuracy. For example, using the same model with constant
inputs produces the following results.
7 In the original model with inputs U and V, select the Optimization > Use
integer division to handle net slopes that are reciprocals of integers
optimization parameter, update diagram, and generate code again.
8 In the generated code, the value of the net slope correction is now the ideal
value of 0.001. Using division, the results are numerically accurate.
37-24
Optimize Your Generated Code
In the model with constant inputs, select the Optimization > Use
integer division to handle net slopes that are reciprocals of
integers optimization parameter and simulate the model.
Note This setting must match your target hardware rounding mode.
• For the Product and Data Type Conversion blocks in the model, the
Integer rounding mode is set to Simplest.
• Model does not use multiword operations.
37-25
37 Code Generation
When you select the Optimize using specified minimum and maximum
values configuration parameter, the software takes into account input
range information, also known as design minimum and maximum, that you
specify for signals and parameters in your model. It uses these minimum and
maximum values to derive range information for downstream signals in the
model and then uses this derived range information to simplify mathematical
operations in the generated code whenever possible.
Prerequisites
The Optimize using specified minimum and maximum values
parameter appears for ERT-based targets only and requires an Embedded
Coder license when generating code.
37-26
Optimize Your Generated Code
2 Specify design minimum and maximum values for signals and parameters
in your model using the tips in “How to Configure Your Model” on page
37-26.
For more information, see “Optimize using the specified minimum and
maximum values”.
Limitations
37-27
37 Code Generation
37-28
Optimize Your Generated Code
fxpdemo_min_max_optimization
5 From the context menu, select C/C++ Code > Navigate To C/C++ Code.
The code generation report highlights the code generated for this block.
The generated code includes a call to the div_repeat_u32 utility function.
6 Click the div_repeat_u32 link to view the utility function, which contains
code for handling division by zero.
37-29
37 Code Generation
1 Double-click the Inport block labelled 5 to open the block parameters dialog
box.
2 On the block parameters dialog box, select the Signal Attributes pane
and note that:
• The Minimum value for this signal is 1.
• The Maximum value for this signal is 100.
8 From the context menu, select C/C++ Code > Navigate To C/C++ Code.
The code generation report highlights the code generated for this block.
This time the generated code implements the division with a shift operation
and there is no division utility function.
37-30
Optimize Your Generated Code
tmp = rtU.In6;
rtY.Out3 = (uint32_T)tmp ==
(uint32_T)0 ? MAX_uint32_T : ((uint32_T)rtU.In5 << 17) /
(uint32_T)tmp;
1 Double-click the Inport block labelled 5 to open the block parameters dialog
box.
2 On the block parameters dialog box, select the Signal Attributes pane and
set the Maximum value to 40000, then click OK to close the dialog box.
5 From the context menu, select C/C++ Code > Navigate To C/C++ Code.
The code generation report highlights the code generated for this block.
The generated code includes a call to the div_repeat_32 utility function.
37-31
37 Code Generation
1 From the Analysis menu of the model you want to analyze, select Model
Advisor.
2 In the Model Advisor left pane, expand the By Product node and select
Embedded Coder.
3 From the Model Advisor Edit menu, select Select All to enable all Model
Advisor checks associated with the selected node. For fixed-point code
generation, the most important check boxes to select are Identify blocks
that generate expensive fixed-point and saturation code, Identify
questionable fixed-point operations, Identify blocks that generate
expensive rounding code, and Check the hardware implementation.
4 Click Run Selected Checks. Any tips for improving the efficiency of your
fixed-point model appear in the Model Advisor window.
The sections that follow discuss fixed-point related checks and sub-checks
found in the Model Advisor. The sections explain the checks, discuss their
importance in fixed-point code generation, and offer suggestions for tweaking
your model to optimize your generated code.
37-32
Optimizing Your Generated Code with the Model Advisor
• When the input range of a Sum block exceeds the output range, a range
error occurs. Users can get any addition or subtraction their application
requires by inserting data type conversion blocks before and/or after the
sum block.
• When a Sum block has an input with a slope adjustment factor that does
not equal the slope adjustment factor of the output, the mismatch requires
the Sum block to perform a multiply operation each time the input is
converted to the output’s data type and scaling. The mismatch can be
removed by changing the scaling of the output or the input.
• When the net sum of the Sum block input biases does not equal the bias of
the output, the generated code includes one extra addition or subtraction
instruction to correctly account for the net bias adjustment. Changing
the bias of the output scaling can make the net bias adjustment zero and
eliminate the need for the extra operation.
• When the input and output of the MinMax block have different data types,
a conversion operation is required every time the block is executed. The
model is more efficient with the same data types.
• When the data type and scaling of the input of the MinMax block does not
match the data type and scaling of the output, a conversion is required
before performing a relational operation. This could result in a range error
when casting, or a precision loss each time a conversion is performed.
Change the scaling of either the input or output to generate more efficient
code.
• When the input of the MinMax block has a different slope adjustment
factor than the output, the MinMax block requires a multiply operation
each time the block is executed to convert the input to the data type and
37-33
37 Code Generation
scaling of the output. You can correct the mismatch by changing the scaling
of either the input or output.
• When the initial condition for the Discrete-Time Integrator blocks is used
to initialize the state and output, the output equation generates excessive
code and an extra global variable is required. It is recommended that you
set the Function Block Parameters > Use initial condition as initial
and reset value for parameter to State only (most efficient).
• If the input data type of the Compare to Zero block cannot represent zero
exactly, the input signal is compared to the closest representable value of
zero, resulting in parameter overflow. To avoid this parameter overflow,
select an input data type that can represent zero.
• If the Compare to Constant block’s Constant value is outside the range
that the input data type can represent, the input signal is compared to
the closest representable value of the constant. This results in parameter
overflow. To avoid this parameter overflow, select an input data type that
can represent the Constant value, or change the Constant value to a
value that can be accommodated by the input data type.
37-34
Optimizing Your Generated Code with the Model Advisor
• Lookup table input data is evenly spaced, but the spacing is not a power of
two.
• You select the Use integer division to handle net slopes that are
reciprocals of integers optimization parameter, but your model
configuration is not compatible with this selection.
• Your model configuration is suitable for using integer division to handle net
slope correction, but you do not select the Use integer division to handle
net slopes that are reciprocals of integers optimization parameter.
For more information, see “Handle Net Slope Correction” on page 37-14.
37-35
37 Code Generation
One such situation is when a calculation using more than one division
operation is computed. A general guideline from the field of numerical
analysis is to multiply all the denominator terms together first, then do
one and only one division. This improves accuracy and often speed in
floating-point and especially fixed-point. This can be accomplished in
Simulink by cascading Product blocks. Note that multiple divisions spread
over a series of blocks are not detected by the Model Advisor.
37-36
Optimizing Your Generated Code with the Model Advisor
37-37
37 Code Generation
trade-offs, the Model Advisor identifies blocks where there is potential for
efficiency improvements, and issues a warning when:
• When the input data types of a Relational Operator block are not the
same, a conversion operation is required every time the block is executed.
If one of the inputs is invariant, then changing the data type and scaling
37-38
Optimizing Your Generated Code with the Model Advisor
of the invariant input to match the other input improves the efficiency
of the model.
• When the inputs of a Relational Operator block have different ranges,
there will be a range error when casting, and a precision loss each time
a conversion is performed. You can insert Data Type Conversion blocks
before the Relational Operator block to convert both inputs to a common
data type that has enough range and precision to represent each input.
• When the inputs of a Relational Operator block have different slope
adjustment factors, the Relational Operator block is required to perform
a multiply operation each time the input with lesser positive range is
converted to the data type and scaling of the input with greater positive
range. The extra multiplication requires extra code, slows down the speed
of execution, and usually introduces additional precision loss. By adjusting
the scaling of the inputs, you can eliminate mismatched slopes.
37-39
37 Code Generation
what you expect from traditional handwritten code, use the simplest rounding
mode. In general the simplest mode provides the minimum cost solution with
no overflows. If the simplest mode is not available, round to floor.
The primary exception to this rule is the rounding behavior of signed integer
division. The C standard leaves this rounding behavior unspecified, but for
most production targets the “no effort” mode is to round to zero. For unsigned
division, everything is nonnegative, so rounding to floor and rounding to zero
are identical. To improve rounding efficiency, set Model Configuration
Parameters > Hardware Implementation > Production hardware >
Signed integer division rounds to using the mode that your production
target uses.
37-40
Use the Model Advisor to Optimize Fixed-Point Operations in Generated Code
37-41
37 Code Generation
37-42
Use the Model Advisor to Optimize Fixed-Point Operations in Generated Code
addpath(fullfile(docroot,'toolbox','fixpoint','examples'))
ex_net_slope5
In this model, the net slope for the Data Type Conversion and Product blocks
is 1/1000.
• The Data type of the two Inport blocks, U and V, is set to int16.
• The Integer rounding mode of the Data Type Conversion block, is set to
Floor, and the Output data type is set to fixdt(1, 33, 1000, 0).
Note Setting the Output data type word length greater than the length
of the long data type results in multiword operations.
1 From the model menu, select Analysis > Model Advisor > Model
Advisor.
2 In the Model Advisor left pane, expand the By Product node and then
expand the Embedded Coder node.
37-43
37 Code Generation
The Model Advisor warns that your model contains a block that will invoke
net slope correction.
a Double click the Data Type Conversion block and change the Integer
rounding mode to Simplest.
b Change the Output data type from fixdt(1,33,1000,0) to
fixdt(1,16,1000,0), to avoid multiword operations.
c Save the model.
The Model Advisor warns that your model contains blocks that will
generate expensive rounding code for multiplication and division. It also
provides recommendations on how to change your model configuration to
make it more efficient.
a Double click on the Product block, and change the Integer rounding
mode to Simplest.
b Save the model.
37-44
Use the Model Advisor to Optimize Fixed-Point Operations in Generated Code
37-45
37 Code Generation
37-46
38
Fixed-Point Advisor
Reference
Fixed-Point Advisor
38-2
Fixed-Point Advisor
Description
Use the Fixed-Point Advisor to:
Procedures
Automatically Run Tasks. The following steps list how you can
automatically run all tasks within a folder.
1 Click the Run to Failure button. The tasks run in order until a task fails.
• Manually fix the problem using the Explore Result button, if present.
• Manually fix the problem by modifying the model as instructed in the
Analysis Result box.
• Automatically fix the problem using the Modify All button, if available.
38-3
38 Fixed-Point Advisor Reference
Run Individual Tasks. The following steps list how you can run an
individual task.
• Manually fix the problem using the Explore Result button, if present.
• Manually fix the problem by modifying the model.
• Automatically fix the problem using the Modify All button, if available.
5 Once you have fixed a Warning or Failed task, rerun the task by clicking
Run This Task.
Run to Selected Task. If you know that a particular task causes a failure,
you might want to run all the tasks prior to this task and save a restore point
before continuing the run. For more information about restore points, see
“Save a Restore Point” on page 30-8. To run all tasks up to and including
the currently selected task:
3 From the context menu, select Run to Selected Task to run all tasks up
to and including the selected task.
Note If a task before the selected task fails, the Fixed-Point Advisor stops
the run at the failed task.
38-4
Fixed-Point Advisor
Rerun a Task. You might want to rerun a task to see if changes you make
result in a different answer. To rerun a task that you have run before:
2 Click the path link listed for Report. A report containing a summary of
all tasks is displayed.
See Also
38-5
38 Fixed-Point Advisor Reference
38-6
Preparing Model for Conversion
Description
Validate model-wide settings and create simulation reference data for
downstream tasks.
See Also
38-7
38 Fixed-Point Advisor Reference
Description
Ensures that fixed-point data can be logged in downstream tasks.
38-8
Preparing Model for Conversion
A model in the model hierarchy is using a Use Modify All or manually configure all
referenced configuration set instead of an models in the model hierarchy to use the active
active configuration set. configuration set. See “Manage a Configuration
Set”.
If you use Modify All, the Fixed-Point
Advisor creates an active configuration set
by copying the referenced configuration set
and configuring the model to use this new
configuration set as the active set.
If there are multiple reference sets, the
Fixed-Point Advisor creates multiple new
configuration sets.
Action Results
Clicking Modify All:
38-9
38 Fixed-Point Advisor Reference
See Also
38-10
Preparing Model for Conversion
Description
A model update diagram action is necessary for most down stream tasks.
See Also
“Update a Block Diagram” in the Simulink documentation
38-11
38 Fixed-Point Advisor Reference
Description
Blocks that do not support fixed-point data types cannot be converted.
38-12
Preparing Model for Conversion
Tips
• Before inserting a replacement block, use the Preview link to view the
replacement block. If necessary, update the settings on the replacement
block to match the settings on the original block.
• If the Fixed-Point Advisor does not recommend a corresponding fixed-point
block, replace the unsupported block with a number of lower-level blocks
to provide the same functionality.
• The goal is to replace all blocks that do not support fixed-point data types.
Using Data Type Conversion blocks to isolate blocks at this stage enables
you to continue running through the conversion process. However, this
will cause the Summarize data type task to fail downstream. To fix
this failure, you must replace the block that does not support fixed-point
data types.
See Also
The Simulink Block Data Type Support table summarizes characteristics of
blocks in the Simulink and Fixed-Point Designer block libraries, including
whether or not they support fixed-point data types. To view this table, enter
the following command at the MATLAB command line:
showblockdatatypetable
38-13
38 Fixed-Point Advisor Reference
Description
The Fixed-Point Advisor uses logged signals to compare the initial data type
to the fixed-point data type.
Tip
Log inports and outports of the system under conversion.
38-14
Preparing Model for Conversion
Description
Simulate the model using the current solver settings, create and archive
reference signal data to use for comparison and analysis in downstream tasks.
Input Parameters
38-15
38 Fixed-Point Advisor Reference
Action Results
Clicking Modify All sets Fixed-point instrumentation mode to Minimums,
maximums and overflows. A table displays the current and previous block
settings.
Tips
• If the simulation is set up to have a long simulation time, after starting the
run of this task you can stop the simulation by selecting the waitbar and
then pressing Ctrl+C. This allows you to change the simulation time and
continue without having to wait for the long simulation to complete.
• Specifying small simulation run times reduces task processing times. You
can change the simulation run time in the Configuration Parameters dialog
box. See “Start time” and “Stop time” in the Simulink reference for more
information.
38-16
Preparing Model for Conversion
Description
Validate model-wide settings.
See Also
38-17
38 Fixed-Point Advisor Reference
Description
If the Model Configuration Parameters > Diagnostic > Data Validity
parameters are set to error, the model update diagram action fails in
downstream tasks.
Action Results
Clicking Modify All sets all Model Configuration Parameters >
Diagnostics > Data Validity > Parameters options to warning. A table
displays the current and previous settings.
38-18
Preparing Model for Conversion
Description
Configuring logic signals as Boolean data optimizes the code generated in
downstream tasks.
Action Results
Clicking Modify All selects the model Model Configuration Parameters >
Optimization > Implement logic signals as Boolean data check box. A
table displays the current and previous parameter settings.
38-19
38 Fixed-Point Advisor Reference
Description
This task identifies:
Note This task is a Simulink task. For more information, see “Check bus
usage” in the Simulink documentation.
38-20
Preparing Model for Conversion
Description
If Model Configuration Parameters > Diagnostics > Simulation range
checking is set to none, the simulation does not generate any range checking
warnings.
Action Results
Clicking Modify All sets Model Configuration Parameters > Diagnostics
> Simulation range checking to warning.
38-21
38 Fixed-Point Advisor Reference
Description
Models with implicit signal resolution attempt to resolve all named signals
and states to Simulink signal objects, which is inefficient and slows
incremental code generation and model reference. This task identifies those
signals and states for which you may turn off implicit signal resolution and
enforce resolution.
Action Results
Clicking Modify All sets Model Configuration Parameters > Diagnostics
> Data Validity > Signal resolution to Explicit only and enforces
resolution for each of the signals and states in the model. Tables display
the current and previous settings.
38-22
Preparing Model for Conversion
See Also
“Resolve Signal Objects for Output Data” in the Simulink documentation
38-23
38 Fixed-Point Advisor Reference
38-24
Preparing for Data Typing and Scaling
Description
The block settings from this folder simplify the initial data typing and scaling.
The optimal block configuration is achieved in later stages. The tasks in this
folder are preparation for automatic data typing.
Tips
Block output and parameter minimum and maximum values can be specified
in this step.
See Also
38-25
38 Fixed-Point Advisor Reference
Description
When blocks have their data types locked, the Fixed-Point Advisor excludes
them from automatic data typing. This task identifies blocks that have locked
data types so that you can unlock them.
Action Results
Clicking Modify All unlocks data types on blocks that currently have locked
data types.
If your model contains referenced models, unlocks data types on blocks for
all referenced models.
38-26
Preparing for Data Typing and Scaling
Description
Inherited data types might lead to data type propagation errors.
Note This task is preparation for automatic data typing, not actual
automatic data typing.
Input Parameters
undefined
int8
uint8
int16
uint16
int32
uint32
fixdt(1,16,4)
38-27
38 Fixed-Point Advisor Reference
38-28
Preparing for Data Typing and Scaling
Action Results
Clicking Modify All explicitly configures the output data types to the
recommended values where possible. Tables list the previous and current
data types for the reconfigured blocks.
38-29
38 Fixed-Point Advisor Reference
Description
Blocks that have input data type constraints might lead to data type
propagation errors.
Note This task is preparation for automatic data typing, not actual
automatic data typing.
Action Results
Clicking Modify All explicitly configures the specified settings to the
recommended value where possible. A table lists the previous and current
settings for the reconfigured blocks.
38-30
Preparing for Data Typing and Scaling
Tip
Removing unnecessary data setting restrictions makes it more likely that the
Propose data types task will succeed downstream.
38-31
38 Fixed-Point Advisor Reference
Description
Identify mismatches between input or output fixed-point data in Stateflow
charts and their counterparts in Simulink models.
Note This task is preparation for automatic data typing, not actual
automatic data typing.
Action Results
Clicking Modify All configures all Stateflow charts to have strong data
typing with Simulink I/O.
If your model contains referenced models, configures this setting for all
referenced models.
38-32
Preparing for Data Typing and Scaling
Description
This task prepares your model for automatic data typing by identifying and
removing redundant data type specification originating from blocks and
Simulink signal objects.
Note You must rerun this task whenever you delete or manipulate a
Simulink signal object in the base workspace.
Input Parameters
Blocks
Identify and remove redundant data type specification from blocks.
Signal objects
Identify and remove redundant data type specification from Simulink
signal objects.
38-33
38 Fixed-Point Advisor Reference
Action Results
Clicking Modify All explicitly configures the properties of the blocks or
Simulink signal objects to the recommended value where possible. A table
displays the current and previous settings.
38-34
Preparing for Data Typing and Scaling
Description
Review the hardware device settings and verify they are the settings you
intend to use.
Input Parameters
Remain floating-point
Use this setting if you are converting only part of the model to fixed
point and want to leave the rest of the model as floating point.
Same as embedded hardware integer
Use this setting if the hardware device specified is a microprocessor.
int8
int16
int32
fixdt(1,16,4)
38-35
38 Fixed-Point Advisor Reference
See Also
• “Device type”
• “Device vendor”
38-36
Preparing for Data Typing and Scaling
Description
Block output and parameter minimum and maximum values are used for
fixed-point data typing in other tasks. Typically, they are determined during
the design process based on the system you are creating.
Note This task is preparation for automatic data typing, not actual
automatic data typing.
Tips
• In this task, you can specify minimum and maximum values for any block.
• You can promote simulation minimum and maximum values to output
minimum and maximum values using the Model Advisor Result Explorer,
launched by clicking the Explore Result button. In the center pane
of the Model Advisor Result Explorer, use the check boxes in the
PromoteSimMinMax column to promote values.
• If you do not specify block minimum and maximum values, the Propose
data types task might fail later in the conversion.
38-37
38 Fixed-Point Advisor Reference
See Also
“Batch-Fix Warnings or Failures” in the Simulink documentation.
38-38
Return to the Fixed-Point Tool to Perform Data Typing and Scaling
See Also
• “Preparation for Fixed-Point Conversion” on page 30-2
• “Converting a Model from Floating- to Fixed-Point Using Simulation Data”
on page 30-11
38-39
38 Fixed-Point Advisor Reference
38-40
39
Troubleshooting
x = fi(16000,1,8)
produces a fixed-point number with the specified signedness and word length,
and best precision fraction length.
x =
16000
39-2
Frequently Asked Questions About Fixed-Point Numbers
disp(bin(x))
01111101
There are seven implicit zeros at the end of this number before the binary
point because the fraction length of x is -7.
bin2dec('011111010000000')
ans =
16000
You can also find the real world value using the equation
Q = storedInteger(x)
Q =
125
real_world_value =
39-3
39 Troubleshooting
16000
x = fi(.0234,1,8)
x =
0.0234
disp(bin(x))
01100000
There are four implied leading zeros before the binary point because the
fraction length of x is four greater than the word length.
bin2dec('000001100000')*2^(-12)
ans =
0.0234
You can also find the real world value using the equation
39-4
Frequently Asked Questions About Fixed-Point Numbers
Q = storedInteger(x)
Q =
96
real_world_value =
0.0234
If this behavior is undesirable for your application, you can do one of the
following:
• Use the cast function to create a fi object using the globalfimath rules.
fi(x,1,16,10,'RoundingMethod','Floor','OverflowAction','Wrap');
Note When you create a fi object using the cast function, the resulting fi
object does not have a local fimath. When you create a fi object with fimath
properties in the constructor, the fi object does have a local fimath.
39-5
39 Troubleshooting
• Manual Workflow
The manual workflow provides the most control to optimize the fixed-point
types, but requires a greater understanding of fixed-point concepts.
For more information, see “Manual Fixed-Point Conversion Best Practices”
on page 8-89.
• Automated Workflow
The automated Fixed-Point Conversion Tool enables you to convert your
MATLAB code to fixed-point code without requiring extensive preexisting
knowledge of fixed-point concepts. However, this workflow provides less
control over your data types.
For more information, see “Automated Fixed-Point Conversion Best
Practices” on page 8-110.
39-6
Decide Which Workflow Is Right For Your Application
39-7
39 Troubleshooting
• RoundingMethod: Nearest
• OverflowAction: Saturate
• ProductMode: FullPrecision
• SumMode: FullPrecision
These settings offer the smallest rounding error and prevent overflows.
However, they can result in extra logic in generated code. For leaner code, it
is recommended that you match the fimath settings to the settings of your
processor.
• The KeepLSB setting for ProductMode and SumMode models the behavior
of integer operations in the C language. KeepMSB for ProductMode models
the behavior of many DSP devices.
• Different rounding methods require different amounts of overhead code.
Setting the RoundingMethod property to Floor, which is equivalent
to two’s complement truncation, provides the most efficient rounding
implementation for most operations. For the divide function, the most
efficient RoundingMethod is Zero
• The standard method for handling overflows is to wrap using modulo
arithmetic. Other overflow handling methods create costly logic. It is
recommended that you set the OverflowAction property to Wrap when
possible.
39-8
Tips for Making Generated Code More Efficient
CORDIC
The CORDIC-based algorithms are among the most hardware friendly
because they require only iterative shift-add operations. Replacing functions
with one of the CORDIC implementations can make your generated code
more efficient. For a list of the CORDIC functions, and examples of them
being implemented, see “CORDIC”.
Lookup tables
You can implement some functions more efficiently by using a lookup table
approach. For an example, see “Implement Fixed-Point Log2 Using Lookup
Table”.
Division
Division is often not supported by hardware. When possible, it is best to
avoid division operations.
When the denominator is a power of two, you can rewrite the division as a
bitshift operation.
x/8
can be rewritten as
bitsra(x,3)
x/5
can be rewritten as
x*0.2
39-9
39 Troubleshooting
39-10
What to Do If a Function Is Not Supported for Fixed-Point Conversion
The following code returns an error because the log function is not supported
for fixed-point inputs.
x = fi(rand(3),1,16,15);
y = log(x)
Cast the input, x, to a double, and then cast the output back to a fixed-point
data type.
y = fi(log(double(x)),1,16)
y =
This casting allows you to continue with your conversion until you can find a
replacement.
39-11
39 Troubleshooting
Polynomial Approximation
You can approximate the results of a function that is not supported for
fixed-point with a polynomial approximation. For an example, see “Calculate
Fixed-Point Arctangent”.
User-Authored Function
You can also write your own function that supports fixed-point inputs. For
example, using the mod function, which does support fixed-point inputs,
you can write your own version of the rem function, which does not support
fixed-point inputs.
39-12
Common Errors and Warnings
fi*non-fi Errors
Original Algorithm New Algorithm
Issue:
function y = myProduct(x) function y = myProduct(x)
y = 1; When multiplying a y = ones(1,1, 'like', x(1)*x(1))
for n=1:length(x) fixed-point variable by a for n=1:length(x)
y(:)=y*x(n); non-fixed-point variable, the y(:)=y*x(n);
end variable that does not have a end
end fixed-point type can only be a end
constant.
Fix:
Before instrumenting your
code, cast the non-fi variable to
an acceptable fixed-point type.
39-13
39 Troubleshooting
39-14
Common Errors and Warnings
39-15
39 Troubleshooting
2 Under the Signal Attributes tab, open the Data Type Assistant by
clicking .
You can set this override to off at the command line by changing the Data
Type Override setting of a signal’s numerictype. The output data type of this
block remains a built-in uint8 even after performing data type override.
39-16
Why Did The Fixed-Point Tool Not Propose Data Types?
If it is not your intent to use an inherited data type, remove the output data
type inheritance of the block to get a data type proposal.
39-17
39 Troubleshooting
39-18
What to Do When a Block Is Not Supported For Fixed-Point Conversion
The subsystem shown is designed for use on an embedded processor and must
be converted to fixed point. The Chirp Signal block creates simulation data.
The Chirp Signal block supports only floating-point double outputs. However,
if you decouple the Chirp Signal from the rest of the model by inserting a data
type converter block after the Chirp Signal block, you can use the Fixed-Point
Tool to continue converting the Subsystem to fixed point.
User-Authored Blocks
You can create your own block which is supported by the Fixed-Point Tool
from one of the blocks in the User-Defined Functions Library.
39-19
39 Troubleshooting
39-20
A
Writing Fixed-Point
S-Functions
• matlabroot/simulink/include/
• matlabroot/toolbox/simulink/fixedandfloat/fxpdemos/
A-2
Data Type Support
- fltu32_S3_B5
For more information, see “Fixed-Point Data Type and Scaling Notation” on
page 27-18.
In either case, trivial scaling means that the real-world value is equal to the
stored integer value:
All integers, including Simulink built-in integers such as uint8, are treated
as fixed-point numbers with trivial scaling by this API. However, Simulink
built-in integers are different in that their use does not cause a Fixed-Point
Designer software license to be checked out.
A-3
A Writing Fixed-Point S-Functions
• A signal that has been overridden with Scaled double is treated as being
of data type ScaledDouble.
For more information about the Fixed-Point Tool and data type override, see
“Fixed-Point Tool” on page 31-2 and the fxptdlg reference page.
A-4
Structure of the S-Function
The callouts in the diagram alert you to the fact that you must include
fixedpoint.h and fixedpoint.c at the appropriate places in the S-function.
The other elements of the S-function displayed in the diagram follow the
standard requirements for S-functions.
A-5
A Writing Fixed-Point S-Functions
A-6
Storage Containers
Storage Containers
In this section...
“Introduction” on page A-7
“Storage Containers in Simulation” on page A-7
“Storage Containers in Code Generation” on page A-10
Introduction
While coding with the API for user-written fixed-point S-functions, it is
important to keep in mind the difference between storage container size,
storage container word length, and signal word length. The sections that
follow discuss the containers used by the API to store signals in simulation
and code generation.
Signal Container
Container Category Word Length Word Length Container Size
FXP_STORAGE_INT8 (signed) 1 to 8 bits 8 bits 1 byte
FXP_STORAGE_UINT8 (unsigned)
FXP_STORAGE_INT16 (signed) 9 to 16 bits 16 bits 2 bytes
FXP_STORAGE_UINT16 (unsigned)
FXP_STORAGE_INT32 (signed) 17 to 32 bits 32 bits 4 bytes
FXP_STORAGE_UINT32 (unsigned)
A-7
A Writing Fixed-Point S-Functions
Signal Container
Container Category Word Length Word Length Container Size
FXP_STORAGE_OTHER_SINGLE_WORD 33 to word Length of long Length of long data
length of long data type type
data type
FXP_STORAGE_MULTIWORD Greater than Multiples of Multiples of length of
the word length length of long long data type to 128
of long data data type to bits
type to 128 bits 128 bits
When the number of bits in the signal word length is less than the size of the
container, the word length bits are always stored in the least significant bits
of the container. The remaining container bits must be sign extended:
• If the data type is unsigned, the sign extension bits must be cleared to zero.
• If the data type is signed, the sign extension bits must be set to one for
strictly negative numbers, and cleared to zero otherwise.
A-8
Storage Containers
The signal and storage container word lengths are returned by the
ssGetDataTypeFxpWordLength and ssGetDataTypeFxpContainWordLen
functions, respectively. The storage container size is returned by the
ssGetDataTypeStorageContainerSize function. The container category
is returned by the ssGetDataTypeStorageContainCat function, which in
addition to those in the table above, can also return the following values.
A-9
A Writing Fixed-Point S-Functions
Emulation
Because it is valuable for rapid prototyping and hardware-in-the-loop testing,
the emulation of smaller signals inside larger containers is supported in code
generation. For example, a 29-bit signal is supported in code generation
if there is a C data type available that has at least 32 bits. The rules for
A-10
Storage Containers
placing a smaller signal into a larger container, and for dealing with the extra
container bits, are the same in code generation as for simulation.
• FixPt_DataTypeNativeType
• FixPt_DataTypeStorageDouble
• FixPt_DataTypeStorageSingle
• FixPt_DataTypeStorageScaledDouble
• FixPt_DataTypeStorageSInt
• FixPt_DataTypeStorageUInt
• FixPt_DataTypeStorageSLong
• FixPt_DataTypeStorageULong
• FixPt_DataTypeStorageSShort
• FixPt_DataTypeStorageUShort
• FixPt_DataTypeStorageMultiword
A-11
A Writing Fixed-Point S-Functions
The remaining TLC functions listed above return TRUE or FALSE depending
on whether a particular standard C data type is used to hold a given
API-registered data type. Note that these functions do not necessarily give
mutually exclusive answers for a given registered data type, due to the fact
that C data types can potentially overlap in size. In C,
One or more of these C data types can be, and very often are, the same size.
A-12
Data Type IDs
In general, the Simulink software assigns data type IDs during model
initialization on a “first come, first served” basis. For example, consider the
generalized schema of a block diagram below.
The Simulink software assigns a data type ID for each output data type in the
diagram in the order it is requested. For simplicity, assume that the order
of request occurs from left to right. Therefore, the output of block A may be
assigned data type ID 13, and the output of block B may be assigned data
type ID 14. The output data type of block C is the same as that of block A, so
the data type ID assigned to the output of block C is also 13. The output of
block D is assigned data type ID 15.
A-13
A Writing Fixed-Point S-Functions
The Simulink software still assigns the data type IDs in the order in which
they are used. Therefore each data type might end up with a different data
type ID. The output of block A is still assigned data type ID 13. The output
of block D is now next in line and is assigned data type ID 14. The output of
block B is assigned data type ID 15. The output data type of block C is still
the same as that of block A, so it is also assigned data type ID 13.
Note Because of the nature of the assignment of data type IDs, you should
always use API functions to extract information from a data type ID about a
data type in your S-function.
A-14
Data Type IDs
functions will return a data type ID. To see an example of a function being
used, go to the file and line indicated in the table.
• SS_DOUBLE
A-15
A Writing Fixed-Point S-Functions
• SS_SINGLE
• SS_INT8
• SS_UINT8
• SS_INT16
• SS_UINT16
• SS_INT32
• SS_UINT32
• SS_BOOLEAN
You do not need to register these data types. If you attempt to register a
built-in data type, the registration function simply returns the preassigned
data type ID.
1 Register a data type using one of the functions listed in the table Data Type
Registration Functions on page A-15. A data type ID is returned to you.
Alternately, you can use one of the preassigned data type IDs of the
Simulink built-in data types.
2 Use the data type ID to set the data type for an input or output port,
run-time parameter, or DWork state using one of the following functions:
• ssSetInputPortDataType
• ssSetOutputPortDataType
• ssSetRunTimeParamInfo
• ssSetDWorkDataType
A-16
Data Type IDs
• ssGetInputPortDataType
• ssGetOutputPortDataType
• ssGetRunTimeParamInfo
• ssGetDWorkDataType
A-17
A Writing Fixed-Point S-Functions
A-18
Data Type IDs
A-19
A Writing Fixed-Point S-Functions
Example of
Token Description Use
FXP_ROUND_CEIL Round to the closest representable number Not Available
in the direction of positive infinity
FXP_ROUND_CONVERGENT Round toward nearest integer with ties Not Available
rounding to nearest even integer
FXP_ROUND_FLOOR Round to the closest representable number Not Available
in the direction of negative infinity
A-20
Overflow Handling and Rounding Methods
Example of
Token Description Use
FXP_ROUND_NEAR Round to the closest representable number, Not Available
with the exact midpoint rounded in the
direction of positive infinity
FXP_ROUND_NEAR_ML Round toward nearest. Ties round toward Not Available
negative infinity for negative numbers,
and toward positive infinity for positive
numbers
FXP_ROUND_SIMPLEST Automatically chooses between round Not Available
toward floor and round toward zero to
produce generated code that is as efficient
as possible
FXP_ROUND_ZERO Round to the closest representable number Not Available
in the direction of zero
• Overflows
• Saturations
• Divide-by-zeros
A-21
A Writing Fixed-Point S-Functions
When a function that accepts a pointer to the logging structure is invoked, the
function initializes the event counts of the structure to zero. The requested
math operations are then carried out. Each time an event is detected, the
appropriate event count is incremented by one.
• OverflowOccurred
• SaturationOccurred
• DivisionByZeroOccurred
A-22
Create MEX-Files
Create MEX-Files
To create a MEX-file for a user-written fixed-point C S-function on either
Windows or UNIX® systems:
mex('sfun_user_fxp_asr.c','-lfixedpoint')
A-23
A Writing Fixed-Point S-Functions
• sfun_user_fxp_asr.c
• sfun_user_fxp_BiasProbe.c
• sfun_user_fxp_const.c
• sfun_user_fxp_ContainWordLenProbe.c
• sfun_user_fxp_dtprop.c
• sfun_user_fxp_FixedExponentProbe.c
• sfun_user_fxp_FracLengthProbe.c
• sfun_user_fxp_FracSlopeProbe.c
• sfun_user_fxp_IsFixedPointProbe.c
• sfun_user_fxp_IsFloatingPointProbe.c
• sfun_user_fxp_IsFxpFltApiCompatProbe.c
• sfun_user_fxp_IsScalingPow2Probe.c
• sfun_user_fxp_IsScalingTrivialProbe.c
• sfun_user_fxp_IsSignedProbe.c
A-24
Fixed-Point S-Function Examples
• sfun_user_fxp_prodsum.c
• sfun_user_fxp_StorageContainCatProbe.c
• sfun_user_fxp_StorageContainSizeProbe.c
• sfun_user_fxp_TotalSlopeProbe.c
• sfun_user_fxp_U32BitRegion.c
• sfun_user_fxp_WordLengthProbe.c
The sections that follow present smaller portions of code that focus on specific
kinds of tasks you might want to perform within your S-function.
For example, suppose you need to know the data type of your input port. To
do this,
In lines 191 and 192, ssGetInputPortDataType is used to get the data type
ID for the two input ports of the S-function:
dataTypeIdU0 = ssGetInputPortDataType( S, 0 );
dataTypeIdU1 = ssGetInputPortDataType( S, 1 );
Further on in the file, the data type IDs are used with API functions to get
information about the input port data types. In lines 205 through 226, a check
is made to see whether the input port data types are single or double:
storageContainerU0 = ssGetDataTypeStorageContainCat( S,
A-25
A Writing Fixed-Point S-Functions
dataTypeIdU0 );
storageContainerU1 = ssGetDataTypeStorageContainCat( S,
dataTypeIdU1 );
if ( storageContainerU0 == FXP_STORAGE_DOUBLE ||
storageContainerU1 == FXP_STORAGE_DOUBLE )
{
/* Doubles take priority over all other rules.
* If either of first two inputs is double,
* then third input is set to double.
*/
dataTypeIdU2Desired = SS_DOUBLE;
}
else if ( storageContainerU0 == FXP_STORAGE_SINGLE ||
storageContainerU1 == FXP_STORAGE_SINGLE )
{
/* Singles take priority over all other rules,
* except doubles.
* If either of first two inputs is single
* then third input is set to single.
*/
dataTypeIdU2Desired = SS_SINGLE;
}
else
In lines 227 through 244, additional API functions are used to get information
about the data types if they are neither single nor double:
{
isSignedU0 = ssGetDataTypeFxpIsSigned( S, dataTypeIdU0 );
isSignedU1 = ssGetDataTypeFxpIsSigned( S, dataTypeIdU1 );
A-26
Fixed-Point S-Function Examples
The functions used above return whether the data types are signed or
unsigned, as well as their word lengths, fractional slopes, exponents, total
slopes, and biases. Together, these quantities give full information about the
fixed-point data types of the input ports.
For example, suppose you want to set the output port data type of your
S-function. To do this,
1 Register a data type by using one of the functions listed in the table Data
Type Registration Functions on page A-15. A data type ID is returned.
Alternately, you can use one of the predefined data type IDs of the Simulink
built-in data types.
A-27
A Writing Fixed-Point S-Functions
V_ISSIGNED,
V_WORDLENGTH,
V_FRACTIONLENGTH,
1 /* true means obey data type override setting for
this subsystem */ );
ssSetOutputPortDataType( S, 0, DataTypeId );
}
else
{
ssSetOutputPortDataType( S, 0, DYNAMICALLY_TYPED );
}
3 Use the data type ID to get the storage container type of the input.
4 Have a case for each input storage container type you want to handle.
Within each case, you will need to perform the following in some way:
• Create a pointer of the correct type according to the storage container,
and cast the original void pointer into the new fully typed pointer (see a
and c).
• You can now store and use the value by dereferencing the new, fully
typed pointer (see b and d).
For example,
A-28
Fixed-Point S-Function Examples
fxpStorageContainerCategory storageContainerU0 =
ssGetDataTypeStorageContainCat( S, dataTypeIdU0 ); (3)
switch ( storageContainerU0 )
{
case FXP_STORAGE_UINT8: (4)
{
const uint8_T *pU8_Properly_Typed_Pointer_To_U0; (a)
pU8_Properly_Typed_Pointer_To_U0 =
(const uint8_T *)pVoidIn; (c)
u8_Stored_Integer_U0 =
*pU8_Properly_Typed_Pointer_To_U0; (d)
pS8_Properly_Typed_Pointer_To_U0 =
(const int8_T *)pVoidIn; (c)
s8_Stored_Integer_U0 =
*pS8_Properly_Typed_Pointer_To_U0; (d)
A-29
A Writing Fixed-Point S-Functions
3 Use the data type ID to get the storage container type of the output.
4 Have a case for each output storage container type you want to handle.
Within each case, you will need to perform the following in some way:
• Create a pointer of the correct type according to the storage container,
and cast the original void pointer into the new fully typed pointer (see a
and c).
• You can now write the value by dereferencing the new, fully typed
pointer (see b and d).
For example,
fxpStorageContainerCategory storageContainerY0 =
ssGetDataTypeStorageContainCat( S,
dataTypeIdY0 ); (3)
switch ( storageContainerY0 )
{
case FXP_STORAGE_UINT8: (4)
{
const uint8_T *pU8_Properly_Typed_Pointer_To_Y0; (a)
A-30
Fixed-Point S-Function Examples
pU8_Properly_Typed_Pointer_To_Y0 =
(const uint8_T *)pVoidOut; (c)
*pU8_Properly_Typed_Pointer_To_Y0 =
u8_Stored_Integer_Y0; (d)
}
break;
pS8_Properly_Typed_Pointer_To_Y0 =
(const int8_T *)pVoidY0; (c)
*pS8_Properly_Typed_Pointer_To_Y0 =
s8_Stored_Integer_Y0; (d)
}
break;
<snip>
A-31
A Writing Fixed-Point S-Functions
#define MDL_SET_INPUT_PORT_DATA_TYPE
static void mdlSetInputPortDataType(SimStruct *S, int port,
DTypeId dataTypeIdInput)
{
if ( isDataTypeSupported( S, dataTypeIdInput ) )
{
DTypeId dataTypeIdOutput;
dataTypeIdOutput = ssRegisterDataTypeFxpBinaryPoint(
S,
ssGetDataTypeFxpIsSigned( S, dataTypeIdInput ), (a)
ssGetDataTypeFxpWordLength( S, dataTypeIdInput ), (b)
ssGetDataTypeFractionLength( S, dataTypeIdInput )
- V_NUM_BITS_TO_SHIFT_RGHT, (c)
0 /* false means do NOT obey data type override
setting for this subsystem */ );
ssSetOutputPortDataType( S, 0, dataTypeIdOutput );
}
}
A-32
API Function Reference
A-33
ssFxpConvert
Arguments S
SimStruct representing an S-function block.
pVoidDest
Pointer to the converted value.
sizeofDest
Size in memory of the converted value.
dataTypeIdDest
Data type ID of the converted value.
pVoidSrc
Pointer to the value you want to convert.
sizeofSrc
Size in memory of the value you want to convert.
dataTypeIdSrc
Data type ID of the value you want to convert.
roundMode
Rounding mode you want to use if a loss of precision is necessary
during the conversion. Possible values are FXP_ROUND_CEIL,
FXP_ROUND_CONVERGENT, FXP_ROUND_FLOOR, FXP_ROUND_NEAR,
FXP_ROUND_NEAR_ML, FXP_ROUND_SIMPLEST and FXP_ROUND_ZERO.
A-34
ssFxpConvert
overflowMode
Overflow mode you want to use if overflow occurs during the
conversion. Possible values are FXP_OVERFLOW_SATURATE and
FXP_OVERFLOW_WRAP.
pFxpOverflowLogs
Pointer to the fixed-point overflow logging structure.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC None
Functions
A-35
ssFxpConvertFromRealWorldValue
Arguments S
SimStruct representing an S-function block.
pVoidDest
Pointer to the converted value.
sizeofDest
Size in memory of the converted value.
dataTypeIdDest
Data type ID of the converted value.
dblRealWorldValue
Double value you want to convert.
roundMode
Rounding mode you want to use if a loss of precision is necessary
during the conversion. Possible values are FXP_ROUND_CEIL,
FXP_ROUND_CONVERGENT, FXP_ROUND_FLOOR, FXP_ROUND_NEAR,
FXP_ROUND_NEAR_ML, FXP_ROUND_SIMPLEST and FXP_ROUND_ZERO.
overflowMode
Overflow mode you want to use if overflow occurs during the
conversion. Possible values are FXP_OVERFLOW_SATURATE and
FXP_OVERFLOW_WRAP.
pFxpOverflowLogs
Pointer to the fixed-point overflow logging structure.
A-36
ssFxpConvertFromRealWorldValue
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC None
Functions
A-37
ssFxpConvertToRealWorldValue
Arguments S
SimStruct representing an S-function block.
pVoidSrc
Pointer to the value you want to convert.
sizeofSrc
Size in memory of the value you want to convert.
dataTypeIdSrc
Data type ID of the value you want to convert.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC None
Functions
A-38
ssFxpGetU32BitRegion
Purpose Return stored integer value for 32-bit region of real, scalar signal
element
Arguments S
SimStruct representing an S-function block.
pVoid
Pointer to the storage container of the real, scalar signal element
in which the 32-bit region of interest resides.
dataTypeId
Data type ID of the registered data type corresponding to the
signal.
regionIndex
Index of the 32-bit region whose stored integer value you want to
retrieve, where 0 accesses the least significant 32-bit region.
Description This function returns the stored integer value in the 32-bit region
specified by regionIndex, associated with the fixed-point data type
designated by dataTypeId. You can use this function with any
fixed-point data type, including those with word sizes less than 32 bits.
If the fixed-point word size is less than 32 bits, the remaining bits are
sign extended.
This function generates an error if dataTypeId represents a
floating-point data type.
To view an example model whose S-functions use the
ssFxpGetU32BitRegion function, at the MATLAB prompt, enter
fxpdemo_sfun_user_U32BitRegion.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
A-39
ssFxpGetU32BitRegion
Languages C
A-40
ssFxpGetU32BitRegionCompliant
Purpose Determine whether S-function is compliant with the U32 bit region
interface
Arguments S
SimStruct representing an S-function block.
result
• 1 if S-function calls ssFxpSetU32BitRegionCompliant to
declare compliance with memory footprint for fixed-point data
types with 33 or more bits
• 0 if S-function does not call ssFxpSetU32BitRegionCompliant
ssFxpSetU32BitRegionCompliant(S,1);
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
A-41
ssFxpGetU32BitRegionCompliant
Languages C
A-42
ssFxpSetU32BitRegion
Purpose Set stored integer value for 32-bit region of real, scalar signal element
Arguments S
SimStruct representing an S-function block.
pVoid
Pointer to the storage container of the real, scalar signal element
in which the 32-bit region of interest resides.
dataTypeId
Data type ID of the registered data type corresponding to the
signal.
regionValue
Stored integer value that you want to assign to a 32-bit region.
regionIndex
Index of the 32-bit region whose stored integer value you want to
set, where 0 accesses the least significant 32-bit region.
Description This function sets regionValue as the stored integer value of the 32-bit
region specified by regionIndex, associated with the fixed-point data
type designated by dataTypeId. You can use this function with any
fixed-point data type, including those with word sizes less than 32
bits. If the fixed-point word size is less than 32 bits, ensure that the
remaining bits are sign extended.
This function generates an error if dataTypeId represents a
floating-point data type, or if the stored integer value that you set is
invalid.
A-43
ssFxpSetU32BitRegion
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
A-44
ssFxpSetU32BitRegionCompliant
Purpose Declare compliance with the U32 bit region interface for fixed-point
data types with 33 or more bits
Arguments S
SimStruct representing an S-function block.
Value
• 1 declare compliance with memory footprint for fixed-point
data types with 33 or more bits.
Description This function declares compliance with the Fixed-Point Designer bit
region interface for data types with 33 or more bits. The memory
footprint for data types with 33 or more bits varies between MATLAB
host platforms and might change between software releases. To
make an S-function robust to memory footprint changes, use the U32
bit region interface. You can use identical source code on different
MATLAB host platforms and with any software release from R2008b.
If the memory footprint changes between releases, you do not have to
recompile U32 bit region compliant S-functions.
To make an S-function U32 bit region compliant, before calling any
other Fixed-Point Designer API function on data with 33 or more bits,
you must call this function as follows:
ssFxpSetU32BitRegionCompliant(S,1);
A-45
ssFxpSetU32BitRegionCompliant
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
A-46
ssGetDataTypeBias
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know the bias.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC FixPt_DataTypeBias
Functions
A-47
ssGetDataTypeFixedExponent
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know the exponent.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
A-48
ssGetDataTypeFixedExponent
TLC FixPt_DataTypeFixedExponent
Functions
A-49
ssGetDataTypeFracSlope
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know the fractional slope.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
A-50
ssGetDataTypeFracSlope
TLC FixPt_DataTypeFracSlope
Functions
A-51
ssGetDataTypeFractionLength
Purpose Return fraction length of registered data type with power-of-two scaling
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know the fraction length.
Description This function returns the fraction length, or the number of bits to the
right of the binary point, of the data type designated by dataTypeId.
This function errors out when ssGetDataTypeIsScalingPow2 returns
FALSE.
This function also errors out when ssGetDataTypeIsFxpFltApiCompat
returns FALSE.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC FixPt_DataTypeFractionLength
Functions
A-52
ssGetDataTypeFxpContainWordLen
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know the container word length.
Description This function returns the word length, in bits, of the storage
container of the fixed-point data type designated by dataTypeId.
This function does not return the size of the storage container or the
word length of the data type. To get the storage container size, use
ssGetDataTypeStorageContainerSize. To get the data type word
length, use ssGetDataTypeFxpWordLength.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
Examples An sfix24_En10 data type has a word length of 24, but is actually
stored in 32 bits during simulation. For this signal,
A-53
ssGetDataTypeFxpContainWordLen
TLC FixPt_DataTypeFxpContainWordLen
Functions
A-54
ssGetDataTypeFxpIsSigned
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered fixed-point data type for which you
want to know whether it is signed.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC FixPt_DataTypeFxpIsSigned
Functions
A-55
ssGetDataTypeFxpWordLength
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered fixed-point data type for which you
want to know the word length.
Description This function returns the word length of the fixed-point data type
designated by dataTypeId. This function does not return the word
length of the container of the data type. To get the container word
length, use ssGetDataTypeFxpContainWordLen:
• If the registered data type is fixed point, this function returns the
total word length including any sign bits, integer bits, and fractional
bits.
• If the registered data type is ScaledDouble, this function returns the
word length of the nonoverridden data type.
• If registered data type is single or double, this function errors out.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
Examples An sfix24_En10 data type has a word length of 24, but is actually
stored in 32 bits during simulation. For this signal,
A-56
ssGetDataTypeFxpWordLength
TLC FixPt_DataTypeFxpWordLength
Functions
A-57
ssGetDataTypeIsFixedPoint
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know whether it is fixed-point.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC FixPt_DataTypeIsFixedPoint
Functions
A-58
ssGetDataTypeIsFloatingPoint
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know whether it is floating-point.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC FixPt_DataTypeIsFloatingPoint
Functions
A-59
ssGetDataTypeIsFxpFltApiCompat
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
determine compatibility with the API for user-written fixed-point
S-functions.
Description This function determines whether the registered data type is supported
by the API for user-written fixed-point S-functions. The supported data
types are all standard Simulink data types, all fixed-point data types,
and data type override data types.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
A-60
ssGetDataTypeIsScalingPow2
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know whether the scaling is strictly power-of-two.
Description This function determines whether the registered data type is scaled
strictly by a power of two. Fixed-point numbers can be represented as
When bias = 0 and fractional slope = 1, the only scaling factor that
remains is a power of two:
A-61
ssGetDataTypeIsScalingPow2
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC FixPt_DataTypeIsScalingPow2
Functions
A-62
ssGetDataTypeIsScalingTrivial
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know whether the scaling is trivial.
Description This function determines whether the scaling of a registered data type
is trivial. In [Slope Bias] representation, fixed-point numbers can be
represented as
In either case, trivial scaling means that the real-world value is simply
equal to the stored integer value:
Scaling is always trivial for pure integers, such as int8, and also for the
true floating-point types single and double.
This function errors out when ssGetDataTypeIsFxpFltApiCompat
returns FALSE.
A-63
ssGetDataTypeIsScalingTrivial
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
TLC FixPt_DataTypeIsScalingTrivial
Functions
A-64
ssGetDataTypeNumberOfChunks
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know the number of chunks in its multiword storage container.
Description This function returns the number of chunks in the multiword storage
container of the fixed-point data type designated by dataTypeId.
This function is valid only for a registered data type whose storage
container uses a multiword representation. You can use the
ssGetDataTypeStorageContainCat function to identify the storage
container category; for multiword storage containers, the function
returns the category FXP_STORAGE_MULTIWORD.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
A-65
ssGetDataTypeStorageContainCat
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know the container category.
Description This function returns the storage container category of the data type
designated by dataTypeId. The container category returned by this
function is used to store input and output signals, run-time parameters,
and DWorks during Simulink simulations.
During simulation, fixed-point signals are held in one of the types of
containers shown in the following table. Therefore in many cases,
signals are represented in containers with more bits than their actual
word length.
Container
Signal Word
Container Category Word Length Length Container Size
FXP_STORAGE_INT8 (signed) 1 to 8 bits 8 bits 1 byte
FXP_STORAGE_UINT8 (unsigned)
FXP_STORAGE_INT16 (signed) 9 to 16 bits 16 bits 2 bytes
FXP_STORAGE_UINT16 (unsigned)
A-66
ssGetDataTypeStorageContainCat
Container
Signal Word
Container Category Word Length Length Container Size
FXP_STORAGE_INT32 (signed) 17 to 32 bits 32 bits 4 bytes
FXP_STORAGE_UINT32 (unsigned)
When the number of bits in the signal word length is less than the size
of the container, the word length bits are always stored in the least
significant bits of the container. The remaining container bits must be
sign extended to fit the bits of the container:
A-67
ssGetDataTypeStorageContainCat
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
• FixPt_DataTypeNativeType
• FixPt_DataTypeStorageDouble
• FixPt_DataTypeStorageSingle
• FixPt_DataTypeStorageScaledDouble
• FixPt_DataTypeStorageSInt
• FixPt_DataTypeStorageUInt
• FixPt_DataTypeStorageSLong
A-68
ssGetDataTypeStorageContainCat
• FixPt_DataTypeStorageULong
• FixPt_DataTypeStorageSShort
• FixPt_DataTypeStorageUShort
A-69
ssGetDataTypeStorageContainerSize
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know the container size.
Description This function returns the storage container size of the data type
designated by dataTypeId. This function returns the same value as
would the sizeof( ) function; it does not return the word length of
either the storage container or the data type. To get the word length of
the storage container, use ssGetDataTypeFxpContainWordLen. To get
the word length of the data type, use ssGetDataTypeFxpWordLength.
The container of the size returned by this function stores input and
output signals, run-time parameters, and DWorks during Simulink
simulations. It is also the appropriate size measurement to pass to
functions like memcpy( ).
This function errors out when ssGetDataTypeIsFxpFltApiCompat
returns FALSE.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
Examples An sfix24_En10 data type has a word length of 24, but is actually
stored in 32 bits during simulation. For this signal,
A-70
ssGetDataTypeStorageContainerSize
TLC FixPt_GetDataTypeStorageContainerSize
Functions
A-71
ssGetDataTypeTotalSlope
Arguments S
SimStruct representing an S-function block.
dataTypeId
Data type ID of the registered data type for which you want to
know the total slope.
This function returns the total slope, rather than the fractional slope, of
the data type designated by dataTypeId. To get the fractional slope,
use ssGetDataTypeFracSlope:
• If the registered data type has trivial scaling, including double and
single data types, the function returns a total slope of 1.
• If the registered data type is ScaledDouble, the function returns the
total slope of the nonoverridden data type. Refer to the examples
below.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
A-72
ssGetDataTypeTotalSlope
Languages C
Examples The data type sfix32_En4 becomes flts32_En4 with data type override.
The total slope returned by this function in either case is 0.0625 (2-4).
The data type ufix16_s7p98 becomes fltu16_s7p98 with data type
override. The total slope returned by this function in either case is 7.98.
TLC FixPt_DataTypeTotalSlope
Functions
A-73
ssLogFixptInstrumentation
Arguments S
SimStruct representing an S-function block.
minValue
Minimum output value that occurred during simulation.
maxValue
Maximum output value that occurred during simulation.
countOverflows
Number of overflows that occurred during simulation.
countSaturations
Number of saturations that occurred during simulation.
countDivisionsByZero
Number of divisions by zero that occurred during simulation.
*pStrName
The string argument is currently unused.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
A-74
ssLogFixptInstrumentation
Languages C
A-75
ssRegisterDataTypeFxpBinaryPoint
Purpose Register fixed-point data type with binary-point-only scaling and return
its data type ID
Arguments S
SimStruct representing an S-function block.
isSigned
TRUE if the data type is signed.
Description This function fully registers a fixed-point data type with the Simulink
software and returns a data type ID. Note that unlike the standard
Simulink function ssRegisterDataType, you do not need to take any
additional registration steps. The data type ID can be used to specify
A-76
ssRegisterDataTypeFxpBinaryPoint
the data types of input and output ports, run-time parameters, and
DWork states. It can also be used with all the standard data type access
methods in simstruc.h, such as ssGetDataTypeSize.
Use this function if you want to register a fixed-point data type with
binary-point-only scaling. Alternatively, you can use one of the other
fixed-point registration functions:
If the registered data type is not one of the Simulink built-in data types,
a Fixed-Point Designer software license is checked out. To prevent a
Fixed-Point Designer software license from being checked out when you
simply open or view a model, protect registration calls with
if (ssGetSimMode(S) != SS_SIMMODE_SIZES_CALL_ONLY )
ssRegisterDataType...
Note Because of the nature of the assignment of data type IDs, you
should always use API functions to extract information from a data type
ID about a data type in your S-function. For more information, refer
to “Data Type IDs” on page A-13.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
A-77
ssRegisterDataTypeFxpBinaryPoint
A-78
ssRegisterDataTypeFxpFSlopeFixExpBias
Purpose Register fixed-point data type with [Slope Bias] scaling specified in
terms of fractional slope, fixed exponent, and bias, and return its data
type ID
Arguments S
SimStruct representing an S-function block.
isSigned
TRUE if the data type is signed.
A-79
ssRegisterDataTypeFxpFSlopeFixExpBias
Description This function fully registers a fixed-point data type with the Simulink
software and returns a data type ID. Note that unlike the standard
Simulink function ssRegisterDataType, you do not need to take any
additional registration steps. The data type ID can be used to specify
the data types of input and output ports, run-time parameters, and
DWork states. It can also be used with all the standard data type access
methods in simstruc.h, such as ssGetDataTypeSize.
Use this function if you want to register a fixed-point data type by
specifying the word length, fractional slope, fixed exponent, and bias.
Alternatively, you can use one of the other fixed-point registration
functions:
If the registered data type is not one of the Simulink built-in data types,
a Fixed-Point Designer software license is checked out. To prevent a
Fixed-Point Designer software license from being checked out when you
simply open or view a model, protect registration calls with
if (ssGetSimMode(S) != SS_SIMMODE_SIZES_CALL_ONLY )
ssRegisterDataType...
A-80
ssRegisterDataTypeFxpFSlopeFixExpBias
Note Because of the nature of the assignment of data type IDs, you
should always use API functions to extract information from a data type
ID about a data type in your S-function. For more information, refer
to “Data Type IDs” on page A-13.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
A-81
ssRegisterDataTypeFxpScaledDouble
Purpose Register scaled double data type with [Slope Bias] scaling specified in
terms of fractional slope, fixed exponent, and bias, and return its data
type ID
Arguments S
SimStruct representing an S-function block.
isSigned
TRUE if the data type is signed.
A-82
ssRegisterDataTypeFxpScaledDouble
Description This function fully registers a fixed-point data type with the Simulink
software and returns a data type ID. Note that unlike the standard
Simulink function ssRegisterDataType, you do not need to take any
additional registration steps. The data type ID can be used to specify
the data types of input and output ports, run-time parameters, and
DWork states. It can also be used with all the standard data type access
methods in simstruc.h, such as ssGetDataTypeSize.
Use this function if you want to register a scaled double data type.
Alternatively, you can use one of the other fixed-point registration
functions:
If the registered data type is not one of the Simulink built-in data types,
a Fixed-Point Designer software license is checked out. To prevent a
Fixed-Point Designer software license from being checked out when you
simply open or view a model, protect registration calls with
if (ssGetSimMode(S) != SS_SIMMODE_SIZES_CALL_ONLY )
ssRegisterDataType...
A-83
ssRegisterDataTypeFxpScaledDouble
Note Because of the nature of the assignment of data type IDs, you
should always use API functions to extract information from a data type
ID about a data type in your S-function. For more information, refer
to “Data Type IDs” on page A-13.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
Languages C
A-84
ssRegisterDataTypeFxpSlopeBias
Purpose Register data type with [Slope Bias] scaling and return its data type ID
Arguments S
SimStruct representing an S-function block.
isSigned
TRUE if the data type is signed.
Description This function fully registers a fixed-point data type with the Simulink
software and returns a data type ID. Note that unlike the standard
A-85
ssRegisterDataTypeFxpSlopeBias
If the registered data type is not one of the Simulink built-in data types,
a Fixed-Point Designer software license is checked out. To prevent a
Fixed-Point Designer software license from being checked out when you
simply open or view a model, protect registration calls with
if (ssGetSimMode(S) != SS_SIMMODE_SIZES_CALL_ONLY )
ssRegisterDataType...
Note Because of the nature of the assignment of data type IDs, you
should always use API functions to extract information from a data type
ID about a data type in your S-function. For more information, refer
to “Data Type IDs” on page A-13.
Requirement To use this function, you must include fixedpoint.h and fixedpoint.c.
For more information, see “Structure of the S-Function” on page A-5.
A-86
ssRegisterDataTypeFxpSlopeBias
Languages C
TLC None.
Functions
A-87