Additional Assembler Examples
=============================
The examples within the explasm directory show various techniques which may
be useful when writing assembler code.
Example utoa1.s shows Integer to String Conversion, but also shows use of
the stack and a recursive function. This example also uses files udiv10.s
and utoatest.c
Example divc.c generates assembler code for dividing by a constant. One
sample of this generated code is used in testing the above utoa1.s example.
Example random.s shows an assembler routine for generating pseudo-random
numbers. This example also uses the file randtest.c to test the example.
Example bytedemo.c shows how byte order reversal can be achieved in just
four instructions.
1 utoa1.s - Integer to String Conversion
This example shows how to:
* convert an integer to a string in ARM assembly language
* use a stack in an ARM assembly language program
* write a recursive function in ARM assembly language
Its dtoa entry point converts a signed integer to a string of decimal
digits (possibly with a leading '-'); its utoa entry point converts an
unsigned integer to a string of decimal digits.
1.1 Algorithm
Converts a signed integer to a decimal string, generates a '-' and negates
the number if it is negative; then converts the remaining unsigned value.
Converts a given unsigned integer to a decimal string by dividing it by 10,
yielding a quotient and a remainder. The remainder is in the range 0-9 and
is used to create the last digit of the decimal representation. If the
quotient is non-zero it is dealt with in the same way as the original
number, creating the leading digits of the decimal representation;
otherwise the process has finished.
1.2 Explanation
On entry, a2 contains the unsigned integer to be converted and a1
addresses a buffer to hold the character representation of it.
On exit, a1 points immediately after the last digit written.
Both the buffer pointer and the original number have to be saved across
the call to udiv10. This could be done by saving the values to memory.
However, it turns out to be more efficient to use two 'variable' registers,
v1 and v2 (which, in turn, have to be saved to memory).
Because utoa calls other functions, it must save its return link address
passed in lr. The function therefore begins by stacking v1, v2 and lr
using STMFD sp!, {v1,v2,lr}.
In the next block of code, a1 and a2 are saved (across the call to udiv10)
in v1 and v2 respectively and the given number (a2) is moved to the first
argument register (a1) before calling udiv10 with a BL Branch with Link)
instruction.
On return from udiv10, 10 times the quotient is subtracted from the
original number (preserved in v2) by two SUB instructions. The remainder
(in v2) is ready to be converted to character form (by adding ASCII '0')
and to be stored into the output buffer.
But first, utoa has to be called to convert the quotient, unless that is
zero. The next four instructions do this, comparing the quotient (in a1)
with 0, moving the quotient to the second argument register (a2) if not
zero, moving the buffer pointer to the first argument/result register
(a1), and calling utoa if the quotient is not zero.
Note that the buffer pointer is moved to a1 unconditionally: if utoa is
called recursively, a1 will be updated but will still identify the next
free buffer location; if utoa is not called recursively, the next free
buffer location is still needed in a1 by the following code which plants
the remainder digit and returns the updated buffer location (via a1).
The remainder (in a2) is converted to character form by adding '0' and is
then stored in the location addressed by a1. A post-incrementing STRB is
used which stores the character and increments the buffer pointer in a
single instruction, leaving the result value in a1.
Finally, the function is exited by restoring the saved values of v1 and
v2 from the stack, loading the stacked link address into pc and popping
the stack using a single multiple-load instruction:
LDMFD sp!, {v1,v2,pc}
1.3 Creating a runnable example
You can run the utoa routine described here under armsd. To do this, you
must assemble the example and the udiv10 function, compile a simple test
harness written in C, and link the resulting objects together to create a
runnable program.
Copy utua1.s, udiv10.s and utuatest.c from directory examples/explasm to
your current working directory. Then issue the following commands to build
and run the program.
armasm utoa1.s -o utoa1.o
armasm udiv10.s -o udiv10.o
armcc -c utoatest.c
armlink utoa1.o udiv10.o utoatest.o -o utoatest
armsd utoatest
The first two armasm commands assemble the utoa function and the udiv10
function, creating object files utoa1.o and udiv10.o.
The armcc command compiles the test harness. The -c flag tells armcc to compile
only (not to link).
The armlink command links the three objects with the ARM C library to create
an executable (here called utoatest), which can be run under armsd.
Alternatively, use the CodeWarrior project file (utoa.mcp) provided.
1.5 Further Notes - Stacks in assembly language
In this example, three words are pushed on to the stack on entry to utoa
and popped off again on exit. By convention, ARM software uses r13,
usually called sp, as a stack pointer pointing to the last-used word of
a downward growing stack (a so-called 'full, descending' stack). However,
this is only a convention and the ARM instruction set supports equally all
four stacking possibilities: FD, FA, ED, EA.
The instruction used to push values on the stack was:
STMFD sp!, {v1, v2, lr}
The action of this instruction is as follows:
* Subtract 4 * number-of-registers from sp
* Store the registers named in {...} in ascending register number
order to memory at [sp], [sp,4], [sp,8] ...
The matching pop instruction was:
LDMFD sp!, {v1, v2, pc}
Its action is:
* Load the registers named in {...} in ascending register number
order from memory at [sp], [sp,4], [sp,8] ...
* Add 4 * number-of-registers to sp.
Many, if not most, register-save requirements in simple assembly language
programs can be met using this approach to stacks.
A more complete treatment of run-time stacks requires a discussion of:
* stack-limit checking (and extension)
* local variables and stack frames
In the utoa program, you must assume the stack is big enough to deal with
the maximum depth of recursion, and in practice this assumption will be
valid. The biggest 32-bit unsigned integer is about four billion, or ten
decimal digits. This means that at most 10 x 3 registers = 120 bytes have
to be stacked. Because the ARM Procedure Call Standard guarantees that
there are at least 256 bytes of stack available when a function is called,
and because we can guess (or know) that udiv10 uses no stack space, we can
be confident that utoa is quite safe if called by an APCS-conforming
caller such as a compiled C test harness.
The stacking technique illustrated here conforms to the ARM Procedure Call
Standard only if the function using it makes no function calls. Since utoa
calls both udiv10 and itself, it really ought to establish a proper stack
frame. If you really want to write functions that can 'plug and play' together
you will have to follow the APCS exactly.
However, when writing a whole program in assembly language you often know
much more than when writing a program fragment for general, robust
service. This allows you to gently break the APCS in the following way:
* Any chain of function/subroutine calls can be considered compatible with
the APCS provided it uses less than 256 bytes of stack space.
So the utoa example is APCS compatible, even though it is not APCS
confo
嵌入式入门-ADS-安装包
需积分: 0 136 浏览量
更新于2024-04-03
1
收藏 124.74MB RAR 举报
嵌入式系统是计算机科学中的一个重要分支,它涉及到在有限的资源下运行特定功能的硬件和软件集成。在这个领域,ADS(ARM Developer Suite)是一个关键的工具,尤其对于初学者来说,它是进入嵌入式开发的门户。ADS是ARM公司推出的一套针对ARM架构的集成开发环境,主要用于编写、编译、调试基于ARM处理器的应用程序。
标题中的“嵌入式入门-ADS-安装包”暗示了这是一个针对初学者的资源,旨在帮助他们设置和配置ADS开发环境。安装ADS是学习和实践嵌入式开发的第一步,因为它提供了编写C或C++代码、生成可执行文件以及调试代码所需的全部工具。
描述中的“结合安装教程食用更佳”提示用户在使用这个安装包时,最好有一个配套的教程来指导,以确保正确无误地完成安装过程。这通常包括下载安装文件,安装过程中可能的依赖管理,以及安装后的环境变量设置等步骤。
在嵌入式开发中,ADS的主要功能包括:
1. **源代码编辑器**:提供了一个集成的文本编辑器,支持语法高亮、自动完成等功能,帮助开发者编写代码。
2. **编译器**:基于GCC(GNU Compiler Collection),用于将源代码转换为机器语言,生成可执行文件。编译器会检查语法错误、类型匹配等问题,并优化代码以提高执行效率。
3. **连接器**:将编译后的对象文件组合成单一的可执行文件,处理符号引用和内存布局。
4. **调试器**:提供强大的调试功能,包括断点设置、单步执行、查看寄存器状态、内存查看等,使得开发者能够深入理解程序的运行过程。
5. **模拟器/目标调试**:ADS允许在主机上进行模拟调试,也可以通过JTAG或SWD接口连接到真实的硬件设备进行目标板调试。
标签中的“嵌入式”和“软件/插件”进一步强调了ADS在软件开发流程中的角色,特别是在嵌入式硬件环境中。作为一个完整的开发工具链,ADS不仅限于软件开发,还与硬件紧密相连,因此熟悉其操作对于理解嵌入式系统的整体工作原理至关重要。
在文件名称列表中看到的“ADS1.2”可能是指ADS的一个特定版本。每个版本都有其特性、改进和兼容性,选择合适的版本对于项目成功至关重要。例如,ADS1.2可能对某些旧版的ARM处理器有更好的支持,或者在特定的操作系统上运行更加稳定。
嵌入式入门者需要理解ADS的基本概念和操作,熟练掌握其安装和使用,以便在实践中构建、测试和优化针对ARM平台的应用。配合安装教程,这个安装包将是开启嵌入式世界的一把钥匙。通过不断地学习和实践,开发者将逐渐精通ADS,从而在嵌入式系统设计和开发中游刃有余。
be_now_future
- 粉丝: 20
- 资源: 1
最新资源
- FactoryIO液位控制PID仿真实验程序 使用简单的梯形图和SCL语言编写,逻辑清晰,通俗易懂,写有注释,起到抛砖引玉的作用,比较适合有动手能力的入门初学者 软件环境: 1、西门子编程软件:TIA
- 车道检测、自适应巡航控制和编队matlab simulink仿真建模模型
- 双馈风机(DFIG)单机无穷大simulink仿真系统,属于基础程序,可以用来研究双馈风机故障时的短路电流,或者crowbar电路的运行,以及双馈风机的电压穿越,也可在上添加模块进行后续仿真 提供相
- AIGC技术变革与数字中国建设的投资策略分析
- 双路直流电机驱动板 驱动芯片l6205 带资料
- RTL8370N? 8口千兆交机 PCB图纸方案资料 和芯片的datasheet 资料包括PCB图、原理图和物料清单和datasheet 图纸已经过量产测试 画图软件AD10
- 使用随机森林RF做分类建模,有训练集和预测集分类准确率图,还有特征重要性排序图,代码内有注释,可学习性强,直接替数据就可以用,想要的可以加好友我
- 解析Midjourney-生成式AI图像龙头
- 2023年AIGC助力电商营销全产业链,覆盖产品推广到售后服务的应用实例
- 两端500kV LCC-HVDc 输电模型(MATLAB搭建) 基于MATLAB搭建的两端LCC- HVD C输电模型,两端交流电网为345KV,直流电压等级为为500KV,直流电流为2000A,直流
- AI对话模型ChatGPT:OpenAI的最新成果与商业化前景-生成式AI技术的应用和发展
- 自己搭建的三相永磁同步电机直接转矩控制(DTC)模型,控制效果非常好,一共有两个,分别是传统DTC控制和基于滑膜控制改进的DTC控制,联系发:两个DTC控制模型+参考资料 (图1和图2是模型总框架,图
- S7-200 MCGS 基于PLC的饮料灌装控制系统
- SiemenS西门子PLC程序,V90伺服脉冲控制轴,扭矩和速度模式,西门子博途V16 PLC人程序,模拟量输入,威伦屏画面,结构化编程
- PV-3Inverter:基于MATLAB Simulink的太阳能光伏并网逆变器仿真模型 其中,光伏MPPT控制采用扰动观测法(P&O法),三相逆变器模块包含PLL锁相环模块 仿真条件:MATL
- 基于MPC 模型预测控制的轨迹跟随,横向控制模型,车道保持,simulink模型 采用二自由度车辆动力学模型,可以自定义车辆参数,自定义目标轨迹,图中为单移线目标轨迹与实际轨迹偏差