ARMv7-A 那些事 - 6.常用汇编指令
 2023.10.07    |      ARMv7-A    |     AilsonJack    |     暂无评论    |     24 views
By: AilsonJack
Date: 2023-10-07
个人博客: http://www.only2fire.com/
<meta charset="UTF-8"/><meta name="viewport" content="width=device-width initial-scale=1"/><link href="https://fonts.googleapis.com/css?family=Open+Sans:400italic,700italic,700,400&subset=latin,latin-ext" rel="stylesheet" type="text/css"/><style type="text/css">html {overflow-x: initial !important;}:root { --bg-color:#ffffff; --text-color:#333333; --select-text-bg-color:#B5D6FC; --select-text-font-color:auto; --monospace:"Lucida Console",Consolas,"Courier",monospace; --title-bar-height:20px; } .mac-os-11 { --title-bar-height:28px; } html { font-size: 14px; background-color: var(--bg-color); color: var(--text-color); font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; -webkit-font-smoothing: antialiased; } h1, h2, h3, h4, h5 { white-space: pre-wrap; } body { margin: 0px; padding: 0px; height: auto; inset: 0px; font-size: 1rem; line-height: 1.42857; overflow-x: hidden; background: inherit; tab-size: 4; } iframe { margin: auto; } a.url { word-break: break-all; } a:active, a:hover { outline: 0px; } .in-text-selection, ::selection { text-shadow: none; background: var(--select-text-bg-color); color: var(--select-text-font-color); } #write { margin: 0px auto; height: auto; width: inherit; word-break: normal; overflow-wrap: break-word; position: relative; white-space: normal; overflow-x: visible; padding-top: 36px; } #write.first-line-indent p { text-indent: 2em; } #write.first-line-indent li p, #write.first-line-indent p * { text-indent: 0px; } #write.first-line-indent li { margin-left: 2em; } .for-image #write { padding-left: 8px; padding-right: 8px; } body.typora-export { padding-left: 30px; padding-right: 30px; } .typora-export .footnote-line, .typora-export li, .typora-export p { white-space: pre-wrap; } .typora-export .task-list-item input { pointer-events: none; } @media screen and (max-width: 500px) { body.typora-export { padding-left: 0px; padding-right: 0px; } #write { padding-left: 20px; padding-right: 20px; } } #write li >figure:last-child { margin-bottom: 0.5rem; } #write ol, #write ul { position: relative; } img { max-width: 100%; vertical-align: middle; image-orientation: from-image; } button, input, select, textarea { color: inherit; font: inherit; } input[type="checkbox"], input[type="radio"] { line-height: normal; padding: 0px; } *, ::after, ::before { box-sizing: border-box; } #write h1, #write h2, #write h3, #write h4, #write h5, #write h6, #write p, #write pre { width: inherit; } #write h1, #write h2, #write h3, #write h4, #write h5, #write h6, #write p { position: relative; } p { line-height: inherit; } h1, h2, h3, h4, h5, h6 { break-after: avoid-page; break-inside: avoid; orphans: 4; } p { orphans: 4; } h1 { font-size: 2rem; } h2 { font-size: 1.8rem; } h3 { font-size: 1.6rem; } h4 { font-size: 1.4rem; } h5 { font-size: 1.2rem; } h6 { font-size: 1rem; } .md-math-block, .md-rawblock, h1, h2, h3, h4, h5, h6, p { margin-top: 1rem; margin-bottom: 1rem; } .hidden { display: none; } .md-blockmeta { color: rgb(204, 204, 204); font-weight: 700; font-style: italic; } a { cursor: pointer; } sup.md-footnote { padding: 2px 4px; background-color: rgba(238, 238, 238, 0.7); color: rgb(85, 85, 85); border-radius: 4px; cursor: pointer; } sup.md-footnote a, sup.md-footnote a:hover { color: inherit; text-transform: inherit; text-decoration: inherit; } #write input[type="checkbox"] { cursor: pointer; width: inherit; height: inherit; } figure { overflow-x: auto; margin: 1.2em 0px; max-width: calc(100% + 16px); padding: 0px; } figure > table { margin: 0px; } thead, tr { break-inside: avoid; break-after: auto; } thead { display: table-header-group; } table { border-collapse: collapse; border-spacing: 0px; width: 100%; overflow: auto; break-inside: auto; text-align: left; } table.md-table td { min-width: 32px; } .CodeMirror-gutters { border-right: 0px; background-color: inherit; } .CodeMirror-linenumber { user-select: none; } .CodeMirror { text-align: left; } .CodeMirror-placeholder { opacity: 0.3; } .CodeMirror pre { padding: 0px 4px; } .CodeMirror-lines { padding: 0px; } div.hr:focus { cursor: none; } #write pre { white-space: pre-wrap; } #write.fences-no-line-wrapping pre { white-space: pre; } #write pre.ty-contain-cm { white-space: normal; } .CodeMirror-gutters { margin-right: 4px; } .md-fences { font-size: 0.9rem; display: block; break-inside: avoid; text-align: left; overflow: visible; white-space: pre; background: inherit; position: relative !important; } .md-fences-adv-panel { width: 100%; margin-top: 10px; text-align: center; padding-top: 0px; padding-bottom: 8px; overflow-x: auto; } #write .md-fences.mock-cm { white-space: pre-wrap; } .md-fences.md-fences-with-lineno { padding-left: 0px; } #write.fences-no-line-wrapping .md-fences.mock-cm { white-space: pre; overflow-x: auto; } .md-fences.mock-cm.md-fences-with-lineno { padding-left: 8px; } .CodeMirror-line, twitterwidget { break-inside: avoid; } svg { break-inside: avoid; } .footnotes { opacity: 0.8; font-size: 0.9rem; margin-top: 1em; margin-bottom: 1em; } .footnotes + .footnotes { margin-top: 0px; } .md-reset { margin: 0px; padding: 0px; border: 0px; outline: 0px; vertical-align: top; background: 0px 0px; text-decoration: none; text-shadow: none; float: none; position: static; width: auto; height: auto; white-space: nowrap; cursor: inherit; -webkit-tap-highlight-color: transparent; line-height: normal; font-weight: 400; text-align: left; box-sizing: content-box; direction: ltr; } li div { padding-top: 0px; } blockquote { margin: 1rem 0px; } li .mathjax-block, li p { margin: 0.5rem 0px; } li blockquote { margin: 1rem 0px; } li { margin: 0px; position: relative; } blockquote > :last-child { margin-bottom: 0px; } blockquote > :first-child, li > :first-child { margin-top: 0px; } .footnotes-area { color: rgb(136, 136, 136); margin-top: 0.714rem; padding-bottom: 0.143rem; white-space: normal; } #write .footnote-line { white-space: pre-wrap; } @media print { body, html { border: 1px solid transparent; height: 99%; break-after: avoid; break-before: avoid; font-variant-ligatures: no-common-ligatures; } #write { margin-top: 0px; border-color: transparent !important; padding-top: 0px !important; padding-bottom: 0px !important; } .typora-export * { -webkit-print-color-adjust: exact; } .typora-export #write { break-after: avoid; } .typora-export #write::after { height: 0px; } .is-mac table { break-inside: avoid; } #write > p:nth-child(1) { margin-top: 0px; } .typora-export-show-outline .typora-export-sidebar { display: none; } figure { overflow-x: visible; } } .footnote-line { margin-top: 0.714em; font-size: 0.7em; } a img, img a { cursor: pointer; } pre.md-meta-block { font-size: 0.8rem; min-height: 0.8rem; white-space: pre-wrap; background: rgb(204, 204, 204); display: block; overflow-x: hidden; } p > .md-image:only-child:not(.md-img-error) img, p > img:only-child { display: block; margin: auto; } #write.first-line-indent p > .md-image:only-child:not(.md-img-error) img { left: -2em; position: relative; } p > .md-image:only-child { display: inline-block; width: 100%; } #write .MathJax_Display { margin: 0.8em 0px 0px; } .md-math-block { width: 100%; } .md-math-block:not(:empty)::after { display: none; } .MathJax_ref { fill: currentcolor; } [contenteditable="true"]:active, [contenteditable="true"]:focus, [contenteditable="false"]:active, [contenteditable="false"]:focus { outline: 0px; box-shadow: none; } .md-task-list-item { position: relative; list-style-type: none; } .task-list-item.md-task-list-item { padding-left: 0px; } .md-task-list-item > input { position: absolute; top: 0px; left: 0px; margin-left: -1.2em; margin-top: calc(1em - 10px); border: none; } .math { font-size: 1rem; } .md-toc { min-height: 3.58rem; position: relative; font-size: 0.9rem; border-radius: 10px; } .md-toc-content { position: relative; margin-left: 0px; } .md-toc-content::after, .md-toc::after { display: none; } .md-toc-item { display: block; color: rgb(65, 131, 196); } .md-toc-item a { text-decoration: none; } .md-toc-inner:hover { text-decoration: underline; } .md-toc-inner { display: inline-block; cursor: pointer; } .md-toc-h1 .md-toc-inner { margin-left: 0px; font-weight: 700; } .md-toc-h2 .md-toc-inner { margin-left: 2em; } .md-toc-h3 .md-toc-inner { margin-left: 4em; } .md-toc-h4 .md-toc-inner { margin-left: 6em; } .md-toc-h5 .md-toc-inner { margin-left: 8em; } .md-toc-h6 .md-toc-inner { margin-left: 10em; } @media screen and (max-width: 48em) { .md-toc-h3 .md-toc-inner { margin-left: 3.5em; } .md-toc-h4 .md-toc-inner { margin-left: 5em; } .md-toc-h5 .md-toc-inner { margin-left: 6.5em; } .md-toc-h6 .md-toc-inner { margin-left: 8em; } } a.md-toc-inner { font-size: inherit; font-style: inherit; font-weight: inherit; line-height: inherit; } .footnote-line a:not(.reversefootnote) { color: inherit; } .reversefootnote { font-family: ui-monospace, sans-serif; } .md-attr { display: none; } .md-fn-count::after { content: "."; } code, pre, samp, tt { font-family: var(--monospace); } kbd { margin: 0px 0.1em; padding: 0.1em 0.6em; font-size: 0.8em; color: rgb(36, 39, 41); background: rgb(255, 255, 255); border: 1px solid rgb(173, 179, 185); border-radius: 3px; box-shadow: rgba(12, 13, 14, 0.2) 0px 1px 0px, rgb(255, 255, 255) 0px 0px 0px 2px inset; white-space: nowrap; vertical-align: middle; } .md-comment { color: rgb(162, 127, 3); opacity: 0.6; font-family: var(--monospace); } code { text-align: left; vertical-align: initial; } a.md-print-anchor { white-space: pre !important; border-width: initial !important; border-style: none !important; border-color: initial !important; display: inline-block !important; position: absolute !important; width: 1px !important; right: 0px !important; outline: 0px !important; background: 0px 0px !important; text-decoration: initial !important; text-shadow: initial !important; } .os-windows.monocolor-emoji .md-emoji { font-family: "Segoe UI Symbol", sans-serif; } .md-diagram-panel > svg { max-width: 100%; } [lang="flow"] svg, [lang="mermaid"] svg { max-width: 100%; height: auto; } [lang="mermaid"] .node text { font-size: 1rem; } table tr th { border-bottom: 0px; } video { max-width: 100%; display: block; margin: 0px auto; } iframe { max-width: 100%; width: 100%; border: none; } .highlight td, .highlight tr { border: 0px; } mark { background: rgb(255, 255, 0); color: rgb(0, 0, 0); } .md-html-inline .md-plain, .md-html-inline strong, mark .md-inline-math, mark strong { color: inherit; } .md-expand mark .md-meta { opacity: 0.3 !important; } mark .md-meta { color: rgb(0, 0, 0); } @media print { .typora-export h1, .typora-export h2, .typora-export h3, .typora-export h4, .typora-export h5, .typora-export h6 { break-inside: avoid; } } .md-diagram-panel .messageText { stroke: none !important; } .md-diagram-panel .start-state { fill: var(--node-fill); } .md-diagram-panel .edgeLabel rect { opacity: 1 !important; } .md-fences.md-fences-math { font-size: 1em; } .md-fences-advanced:not(.md-focus) { padding: 0px; white-space: nowrap; border: 0px; } .md-fences-advanced:not(.md-focus) { background: inherit; } .typora-export-show-outline .typora-export-content { max-width: 1440px; margin: auto; display: flex; flex-direction: row; } .typora-export-sidebar { width: 300px; font-size: 0.8rem; margin-top: 80px; margin-right: 18px; } .typora-export-show-outline #write { --webkit-flex:2; flex: 2 1 0%; } .typora-export-sidebar .outline-content { position: fixed; top: 0px; max-height: 100%; overflow: hidden auto; padding-bottom: 30px; padding-top: 60px; width: 300px; } @media screen and (max-width: 1024px) { .typora-export-sidebar, .typora-export-sidebar .outline-content { width: 240px; } } @media screen and (max-width: 800px) { .typora-export-sidebar { display: none; } } .outline-content li, .outline-content ul { margin-left: 0px; margin-right: 0px; padding-left: 0px; padding-right: 0px; list-style: none; overflow-wrap: anywhere; } .outline-content ul { margin-top: 0px; margin-bottom: 0px; } .outline-content strong { font-weight: 400; } .outline-expander { width: 1rem; height: 1.42857rem; position: relative; display: table-cell; vertical-align: middle; cursor: pointer; padding-left: 4px; } .outline-expander::before { content: ""; position: relative; font-family: Ionicons; display: inline-block; font-size: 8px; vertical-align: middle; } .outline-item { padding-top: 3px; padding-bottom: 3px; cursor: pointer; } .outline-expander:hover::before { content: ""; } .outline-h1 > .outline-item { padding-left: 0px; } .outline-h2 > .outline-item { padding-left: 1em; } .outline-h3 > .outline-item { padding-left: 2em; } .outline-h4 > .outline-item { padding-left: 3em; } .outline-h5 > .outline-item { padding-left: 4em; } .outline-h6 > .outline-item { padding-left: 5em; } .outline-label { cursor: pointer; display: table-cell; vertical-align: middle; text-decoration: none; color: inherit; } .outline-label:hover { text-decoration: underline; } .outline-item:hover { border-color: rgb(245, 245, 245); background-color: var(--item-hover-bg-color); } .outline-item:hover { margin-left: -28px; margin-right: -28px; border-left: 28px solid transparent; border-right: 28px solid transparent; } .outline-item-single .outline-expander::before, .outline-item-single .outline-expander:hover::before { display: none; } .outline-item-open > .outline-item > .outline-expander::before { content: ""; } .outline-children { display: none; } .info-panel-tab-wrapper { display: none; } .outline-item-open > .outline-children { display: block; } .typora-export .outline-item { padding-top: 1px; padding-bottom: 1px; } .typora-export .outline-item:hover { margin-right: -8px; border-right: 8px solid transparent; } .typora-export .outline-expander::before { content: "+"; font-family: inherit; top: -1px; } .typora-export .outline-expander:hover::before, .typora-export .outline-item-open > .outline-item > .outline-expander::before { content: "−"; } .typora-export-collapse-outline .outline-children { display: none; } .typora-export-collapse-outline .outline-item-open > .outline-children, .typora-export-no-collapse-outline .outline-children { display: block; } .typora-export-no-collapse-outline .outline-expander::before { content: "" !important; } .typora-export-show-outline .outline-item-active > .outline-item .outline-label { font-weight: 700; } .md-inline-math-container mjx-container { zoom: 0.95; } mjx-container { break-inside: avoid; } .CodeMirror { height: auto; } .CodeMirror.cm-s-inner { background: inherit; } .CodeMirror-scroll { overflow: auto hidden; z-index: 3; } .CodeMirror-gutter-filler, .CodeMirror-scrollbar-filler { background-color: rgb(255, 255, 255); } .CodeMirror-gutters { border-right: 1px solid rgb(221, 221, 221); background: inherit; white-space: nowrap; } .CodeMirror-linenumber { padding: 0px 3px 0px 5px; text-align: right; color: rgb(153, 153, 153); } .cm-s-inner .cm-keyword { color: rgb(119, 0, 136); } .cm-s-inner .cm-atom, .cm-s-inner.cm-atom { color: rgb(34, 17, 153); } .cm-s-inner .cm-number { color: rgb(17, 102, 68); } .cm-s-inner .cm-def { color: rgb(0, 0, 255); } .cm-s-inner .cm-variable { color: rgb(0, 0, 0); } .cm-s-inner .cm-variable-2 { color: rgb(0, 85, 170); } .cm-s-inner .cm-variable-3 { color: rgb(0, 136, 85); } .cm-s-inner .cm-string { color: rgb(170, 17, 17); } .cm-s-inner .cm-property { color: rgb(0, 0, 0); } .cm-s-inner .cm-operator { color: rgb(152, 26, 26); } .cm-s-inner .cm-comment, .cm-s-inner.cm-comment { color: rgb(170, 85, 0); } .cm-s-inner .cm-string-2 { color: rgb(255, 85, 0); } .cm-s-inner .cm-meta { color: rgb(85, 85, 85); } .cm-s-inner .cm-qualifier { color: rgb(85, 85, 85); } .cm-s-inner .cm-builtin { color: rgb(51, 0, 170); } .cm-s-inner .cm-bracket { color: rgb(153, 153, 119); } .cm-s-inner .cm-tag { color: rgb(17, 119, 0); } .cm-s-inner .cm-attribute { color: rgb(0, 0, 204); } .cm-s-inner .cm-header, .cm-s-inner.cm-header { color: rgb(0, 0, 255); } .cm-s-inner .cm-quote, .cm-s-inner.cm-quote { color: rgb(0, 153, 0); } .cm-s-inner .cm-hr, .cm-s-inner.cm-hr { color: rgb(153, 153, 153); } .cm-s-inner .cm-link, .cm-s-inner.cm-link { color: rgb(0, 0, 204); } .cm-negative { color: rgb(221, 68, 68); } .cm-positive { color: rgb(34, 153, 34); } .cm-header, .cm-strong { font-weight: 700; } .cm-del { text-decoration: line-through; } .cm-em { font-style: italic; } .cm-link { text-decoration: underline; } .cm-error { color: red; } .cm-invalidchar { color: red; } .cm-constant { color: rgb(38, 139, 210); } .cm-defined { color: rgb(181, 137, 0); } div.CodeMirror span.CodeMirror-matchingbracket { color: rgb(0, 255, 0); } div.CodeMirror span.CodeMirror-nonmatchingbracket { color: rgb(255, 34, 34); } .cm-s-inner .CodeMirror-activeline-background { background: inherit; } .CodeMirror { position: relative; overflow: hidden; } .CodeMirror-scroll { height: 100%; outline: 0px; position: relative; box-sizing: content-box; background: inherit; } .CodeMirror-sizer { position: relative; } .CodeMirror-gutter-filler, .CodeMirror-hscrollbar, .CodeMirror-scrollbar-filler, .CodeMirror-vscrollbar { position: absolute; z-index: 6; display: none; outline: 0px; } .CodeMirror-vscrollbar { right: 0px; top: 0px; overflow: hidden; } .CodeMirror-hscrollbar { bottom: 0px; left: 0px; overflow: auto hidden; } .CodeMirror-scrollbar-filler { right: 0px; bottom: 0px; } .CodeMirror-gutter-filler { left: 0px; bottom: 0px; } .CodeMirror-gutters { position: absolute; left: 0px; top: 0px; padding-bottom: 10px; z-index: 3; overflow-y: hidden; } .CodeMirror-gutter { white-space: normal; height: 100%; box-sizing: content-box; padding-bottom: 30px; margin-bottom: -32px; display: inline-block; } .CodeMirror-gutter-wrapper { position: absolute; z-index: 4; background: 0px 0px !important; border: none !important; } .CodeMirror-gutter-background { position: absolute; top: 0px; bottom: 0px; z-index: 4; } .CodeMirror-gutter-elt { position: absolute; cursor: default; z-index: 4; } .CodeMirror-lines { cursor: text; } .CodeMirror pre { border-radius: 0px; border-width: 0px; background: 0px 0px; font-family: inherit; font-size: inherit; margin: 0px; white-space: pre; overflow-wrap: normal; color: inherit; z-index: 2; position: relative; overflow: visible; } .CodeMirror-wrap pre { overflow-wrap: break-word; white-space: pre-wrap; word-break: normal; } .CodeMirror-code pre { border-right: 30px solid transparent; width: fit-content; } .CodeMirror-wrap .CodeMirror-code pre { border-right: none; width: auto; } .CodeMirror-linebackground { position: absolute; inset: 0px; z-index: 0; } .CodeMirror-linewidget { position: relative; z-index: 2; overflow: auto; } .CodeMirror-wrap .CodeMirror-scroll { overflow-x: hidden; } .CodeMirror-measure { position: absolute; width: 100%; height: 0px; overflow: hidden; visibility: hidden; } .CodeMirror-measure pre { position: static; } .CodeMirror div.CodeMirror-cursor { position: absolute; visibility: hidden; border-right: none; width: 0px; } .CodeMirror div.CodeMirror-cursor { visibility: hidden; } .CodeMirror-focused div.CodeMirror-cursor { visibility: inherit; } .cm-searching { background: rgba(255, 255, 0, 0.4); } span.cm-underlined { text-decoration: underline; } span.cm-strikethrough { text-decoration: line-through; } .cm-tw-syntaxerror { color: rgb(255, 255, 255); background-color: rgb(153, 0, 0); } .cm-tw-deleted { text-decoration: line-through; } .cm-tw-header5 { font-weight: 700; } .cm-tw-listitem:first-child { padding-left: 10px; } .cm-tw-box { border-style: solid; border-right-width: 1px; border-bottom-width: 1px; border-left-width: 1px; border-color: inherit; border-top-width: 0px !important; } .cm-tw-underline { text-decoration: underline; } @media print { .CodeMirror div.CodeMirror-cursor { visibility: hidden; } } :root { --side-bar-bg-color: #fafafa; --control-text-color: #777; } @include-when-export url(https://fonts.googleapis.com/css?family=Open+Sans:400italic,700italic,700,400&subset=latin,latin-ext); /* open-sans-regular - latin-ext_latin */ /* open-sans-italic - latin-ext_latin */ /* open-sans-700 - latin-ext_latin */ /* open-sans-700italic - latin-ext_latin */ html { font-size: 16px; -webkit-font-smoothing: antialiased; } body { font-family: "Open Sans","Clear Sans", "Helvetica Neue", Helvetica, Arial, 'Segoe UI Emoji', sans-serif; color: rgb(51, 51, 51); line-height: 1.6; } #write { max-width: 860px; margin: 0 auto; padding: 30px; padding-bottom: 100px; } @media only screen and (min-width: 1400px) { #write { max-width: 1024px; } } @media only screen and (min-width: 1800px) { #write { max-width: 1200px; } } #write > ul:first-child, #write > ol:first-child{ margin-top: 30px; } a { color: #4183C4; } h1, h2, h3, h4, h5, h6 { position: relative; margin-top: 1rem; margin-bottom: 1rem; font-weight: bold; line-height: 1.4; cursor: text; } h1:hover a.anchor, h2:hover a.anchor, h3:hover a.anchor, h4:hover a.anchor, h5:hover a.anchor, h6:hover a.anchor { text-decoration: none; } h1 tt, h1 code { font-size: inherit; } h2 tt, h2 code { font-size: inherit; } h3 tt, h3 code { font-size: inherit; } h4 tt, h4 code { font-size: inherit; } h5 tt, h5 code { font-size: inherit; } h6 tt, h6 code { font-size: inherit; } h1 { font-size: 2.25em; line-height: 1.2; border-bottom: 1px solid #eee; } h2 { font-size: 1.75em; line-height: 1.225; border-bottom: 1px solid #eee; } /*@media print { .typora-export h1, .typora-export h2 { border-bottom: none; padding-bottom: initial; } .typora-export h1::after, .typora-export h2::after { content: ""; display: block; height: 100px; margin-top: -96px; border-top: 1px solid #eee; } }*/ h3 { font-size: 1.5em; line-height: 1.43; } h4 { font-size: 1.25em; } h5 { font-size: 1em; } h6 { font-size: 1em; color: #777; } p, blockquote, ul, ol, dl, table{ margin: 0.8em 0; } li>ol, li>ul { margin: 0 0; } hr { height: 2px; padding: 0; margin: 16px 0; background-color: #e7e7e7; border: 0 none; overflow: hidden; box-sizing: content-box; } li p.first { display: inline-block; } ul, ol { padding-left: 30px; } ul:first-child, ol:first-child { margin-top: 0; } ul:last-child, ol:last-child { margin-bottom: 0; } blockquote { border-left: 4px solid #dfe2e5; padding: 0 15px; color: #777777; } blockquote blockquote { padding-right: 0; } table { padding: 0; word-break: initial; } table tr { border: 1px solid #dfe2e5; margin: 0; padding: 0; } table tr:nth-child(2n), thead { background-color: #f8f8f8; } table th { font-weight: bold; border: 1px solid #dfe2e5; border-bottom: 0; margin: 0; padding: 6px 13px; } table td { border: 1px solid #dfe2e5; margin: 0; padding: 6px 13px; } table th:first-child, table td:first-child { margin-top: 0; } table th:last-child, table td:last-child { margin-bottom: 0; } .CodeMirror-lines { padding-left: 4px; } .code-tooltip { box-shadow: 0 1px 1px 0 rgba(0,28,36,.3); border-top: 1px solid #eef2f2; } .md-fences, code, tt { border: 1px solid #e7eaed; background-color: #f8f8f8; border-radius: 3px; padding: 0; padding: 2px 4px 0px 4px; font-size: 0.9em; } code { background-color: #f3f4f4; padding: 0 2px 0 2px; } .md-fences { margin-bottom: 15px; margin-top: 15px; padding-top: 8px; padding-bottom: 6px; } .md-task-list-item > input { margin-left: -1.3em; } @media print { html { font-size: 13px; } pre { page-break-inside: avoid; word-wrap: break-word; } } .md-fences { background-color: #f8f8f8; } #write pre.md-meta-block { padding: 1rem; font-size: 85%; line-height: 1.45; background-color: #f7f7f7; border: 0; border-radius: 3px; color: #777777; margin-top: 0 !important; } .mathjax-block>.code-tooltip { bottom: .375rem; } .md-mathjax-midline { background: #fafafa; } #write>h3.md-focus:before{ left: -1.5625rem; top: .375rem; } #write>h4.md-focus:before{ left: -1.5625rem; top: .285714286rem; } #write>h5.md-focus:before{ left: -1.5625rem; top: .285714286rem; } #write>h6.md-focus:before{ left: -1.5625rem; top: .285714286rem; } .md-image>.md-meta { /*border: 1px solid #ddd;*/ border-radius: 3px; padding: 2px 0px 0px 4px; font-size: 0.9em; color: inherit; } .md-tag { color: #a7a7a7; opacity: 1; } .md-toc { margin-top:20px; padding-bottom:20px; } .sidebar-tabs { border-bottom: none; } #typora-quick-open { border: 1px solid #ddd; background-color: #f8f8f8; } #typora-quick-open-item { background-color: #FAFAFA; border-color: #FEFEFE #e5e5e5 #e5e5e5 #eee; border-style: solid; border-width: 1px; } /** focus mode */ .on-focus-mode blockquote { border-left-color: rgba(85, 85, 85, 0.12); } header, .context-menu, .megamenu-content, footer{ font-family: "Segoe UI", "Arial", sans-serif; } .file-node-content:hover .file-node-icon, .file-node-content:hover .file-node-open-state{ visibility: visible; } .mac-seamless-mode #typora-sidebar { background-color: #fafafa; background-color: var(--side-bar-bg-color); } .md-lang { color: #b4654d; } /*.html-for-mac { --item-hover-bg-color: #E6F0FE; }*/ #md-notification .btn { border: 0; } .dropdown-menu .divider { border-color: #e5e5e5; opacity: 0.4; } .ty-preferences .window-content { background-color: #fafafa; } .ty-preferences .nav-group-item.active { color: white; background: #999; } .menu-item-container a.menu-style-btn { background-color: #f5f8fa; background-image: linear-gradient( 180deg , hsla(0, 0%, 100%, 0.8), hsla(0, 0%, 100%, 0)); }</style><title>ARMv7-A 那些事 - 6.常用汇编指令</title><p>对于搞嵌入式驱动或者操作系统的人来说,掌握汇编语言的使用还是比较重要的,毕竟有时候在分析定位问题的时候,多多少少都会有汇编的身影。本文主要讲讲ARM指令集格式以及常用的ARM汇编指令(主要包括<code>LDR</code>和<code>STR</code>指令,<code>LDM</code>和<code>STM</code>指令,<code>push</code>和<code>pop</code>指令,<code>MOV</code>指令,<code>CPS</code>指令,<code>MRS</code>和<code>MSR</code>指令,<code>MRC</code>和<code>MCR</code>指令,其余指令暂时没列出来,用到时可以查看ARM手册进行了解)。</p><h2 id="arm指令集格式">ARM指令集格式</h2><p>ARMv7架构是一个32位的处理器架构。同时ARM架构是一个加载/存储体系结构,所有的数据处理操作需要在通用寄存器中完成。</p><p>要学习了解处理器的汇编指令,那么首先可以看看汇编指令的通用表达式,具体的指令也就是使用具体的指令和参数代替通用表达式的参数。ARM指令集的指令表达如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">opcode{&lt;cond&gt;}{S}&nbsp;&lt;Rd&gt;,&nbsp;&lt;Rn&gt;&nbsp;{,&nbsp;&lt;Rm&gt;}</pre><p><code>opcode</code>:指令助记符,比如LDR,STR,MOV等。</p><p><code>{}</code>:大括号括起来的内容表示可选。</p><p><code>&lt;&gt;</code>:<code>&lt;&gt;</code>括起来的内容是必须的。</p><p><code>cond</code>:条件码,比如EQ,NE,CS等,条件码的内容如下图所示:</p><p style="text-align:center"><img src="/uploads/AilsonJack/2023.10.07/143156815106298.png" onclick="preview_image(&#39;/uploads/AilsonJack/2023.10.07/143156815106298.png&#39;)"/></p><p style="text-align:center"><img src="/uploads/AilsonJack/2023.10.07/143157526053255.png" onclick="preview_image(&#39;/uploads/AilsonJack/2023.10.07/143157526053255.png&#39;)"/></p><p><code>S</code>:可选的后缀,如果指令中添加了<code>S</code>,那么指令的执行结果将会影响到CPSR寄存器的标志位域。</p><p><code>Rd</code>:目标寄存器。</p><p><code>Rn</code>:第一个操作数寄存器。</p><p><code>Rm</code>:第二个操作数寄存器。</p><p>在了解了ARM指令的表达式之后,下面就讲讲常用的汇编指令。</p><h2 id="ldr和str指令">LDR和STR指令</h2><p><code>LDR</code>指令用于从内存中读取数据存储到通用寄存器中。<code>STR</code>指令用于将通用寄存器中的值存储到内存中。<code>LDR</code>指令的语法如下所示:</p><pre class="brush:cpp;toolbar:false;" spellcheck="false" lang="asm">LDR{type}{T}{cond}&nbsp;Rt,&nbsp;[Rn&nbsp;{,&nbsp;#offset}] LDR{type}{cond}&nbsp;Rt,&nbsp;[Rn,&nbsp;#offset]! LDR{type}{T}{cond}&nbsp;Rt,&nbsp;[Rn],&nbsp;#offset LDR{type}{cond}&nbsp;Rt,&nbsp;[Rn,&nbsp;+/-Rm] LDR{type}{cond}&nbsp;Rt,&nbsp;[Rn,&nbsp;+/-Rm]! LDR{type}{T}{cond}&nbsp;Rt,&nbsp;[Rn],&nbsp;+/-Rm</pre><p><code>STR</code>指令的语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">STR{type}{T}{cond}&nbsp;Rt,&nbsp;[Rn&nbsp;{,&nbsp;#offset}] STR{type}{cond}&nbsp;Rt,&nbsp;[Rn,&nbsp;#offset]! STR{type}{T}{cond}&nbsp;Rt,&nbsp;[Rn],&nbsp;#offset STR{type}{cond}&nbsp;Rt,&nbsp;[Rn,&nbsp;+/-Rm] STR{type}{cond}&nbsp;Rt,&nbsp;[Rn,&nbsp;+/-Rm]! STR{type}{T}{cond}&nbsp;Rt,&nbsp;[Rn],&nbsp;+/-Rm</pre><p><code>type</code>:操作的数据宽度,可以是:<code>B</code>(unsigned byte),<code>SB</code>(signed byte),<code>H</code>(unsigned halfword),<code>SH</code>(signed halfword)。</p><p><code>cond</code>:条件码。</p><p><code>Rt</code>:目标寄存器。</p><p><code>Rn</code>:存储内存操作基地址的寄存器。</p><p><code>Rm</code>:存储偏移量的寄存器。</p><p><code>offset</code>:立即数。</p><p><code>!</code>:如果存在,表示最终的地址要写回<code>Rn</code>。</p><p><code>T</code>:表示处理器是在用户模式下访问内存地址。</p><p>加载存储指令有4种寻址方式,LDR的操作描述如下(STR指令的操作类似):</p><ul class=" list-paddingleft-2"><li><p>寄存器寻址:要寻址的地址存放在寄存器中。</p></li><li><p>前变基寻址:在内存访问之前,将寄存器中的内存地址加上偏移量之后作为新的内存地址进行内存访问。指令形式为:<code>LDR Rt, [Rn, Op2]</code>。偏移量<code>Op2</code>可以是正数或者是负数,可以是一个立即数,可以是另一个寄存器的值,可以是另一个寄存器中的数据进行移位之后的值。</p></li><li><p>带写回的前变基寻址:指令形式为:<code>LDR Rt, [Rn, Op2]!</code>。该寻址模式和前变基寻址一样,只是在访问完内存之后Rn寄存器中的值就更新为运算之后得到的新内存地址的值。</p></li><li><p>带写回的后变基寻址:指令形式为:<code>LDR Rt, [Rn], #offset</code>和<code>LDR Rt, [Rn], +/-Rm</code>。将寄存器Rn中存储的数值作为内存地址,将该内存地址中的数据读出来存储到Rt寄存器中,然后将内存地址加减立即数offset或者Rm寄存器中的数值得到新的内存地址存储到Rn寄存器中。</p></li></ul><p>上面说的这些有可能不太好懂,下面简单的列举几个例子吧:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm" style="break-inside: unset;">#&nbsp;寄存器寻址 #&nbsp;将R1地址处的数据读出,保存到R0中 LDR&nbsp;R0,&nbsp;[R1] #&nbsp;前变基寻址 #&nbsp;将(R1+0x4)地址处的数据读出,保存到R0中 LDR&nbsp;R0,&nbsp;[R1,&nbsp;#0x4] #&nbsp;将(R1-0x4)地址处的数据读出,保存到R0中 LDR&nbsp;R0,&nbsp;[R1,&nbsp;#-0x4] #&nbsp;将(R1+R2)地址处的数据读出,保存到R0中 LDR&nbsp;R0,&nbsp;[R1,&nbsp;R2] #&nbsp;将((R1+(R2&lt;&lt;2))地址处的数据读出,保存到R0中 LDR&nbsp;R0,&nbsp;[R1,&nbsp;R2,&nbsp;LSL&nbsp;#2] #&nbsp;带写回的前变基寻址 #&nbsp;将(R1+R2)地址处的数据读出,保存到R0中,然后更新R1=R1+R2 LDR&nbsp;R0,&nbsp;[R1,&nbsp;R2]! #&nbsp;将((R1+(R2&lt;&lt;2))地址处的数据读出,保存到R0中,然后更新R1=((R1+(R2&lt;&lt;2)) LDR&nbsp;R0,&nbsp;[R1,&nbsp;R2,&nbsp;LSL&nbsp;#2]! #&nbsp;带写回的后变基寻址 #&nbsp;将R1地址处的数据读出,保存到R0中,然后更新R1=R1+0x4 LDR&nbsp;R0,&nbsp;[R1],&nbsp;#0x4 #&nbsp;将R1地址处的数据读出,保存到R0中,然后更新R1=R1-0x4 LDR&nbsp;R0,&nbsp;[R1],&nbsp;#-0x4 #&nbsp;将R1地址处的数据读出,保存到R0中,然后更新R1=R1+R2 LDR&nbsp;R0,&nbsp;[R1],&nbsp;R2 #&nbsp;将R1地址处的数据读出,保存到R0中,然后更新R1=R1-R2 LDR&nbsp;R0,&nbsp;[R1],&nbsp;-R2</pre><p>STR指令的操作和LDR指令类似,这里就不列举了。</p><h3 id="ldr伪指令">LDR伪指令</h3><p><code>LDR</code>相关的伪指令语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;将数据&nbsp;constant&nbsp;加载到&nbsp;Rt&nbsp;寄存器中 LDR&nbsp;Rt,&nbsp;=constant #&nbsp;将&nbsp;label&nbsp;所代表的地址加载到&nbsp;Rt&nbsp;寄存器中 LDR&nbsp;Rt,&nbsp;=label</pre><p>下面是LDR伪指令简单的使用:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;将&nbsp;0xaa&nbsp;加载到&nbsp;R0&nbsp;寄存器 LDR&nbsp;R0,&nbsp;=0xaa #&nbsp;将&nbsp;_start&nbsp;所代表的地址加载到&nbsp;R0&nbsp;寄存器 LDR&nbsp;R0,&nbsp;=_start</pre><h2 id="ldm和stm指令">LDM和STM指令</h2><p><code>LDM</code>指令用于加载指定地址上的数据保存到一个或者多个寄存器中。<code>STM</code>指令用于将一个或者多个寄存器中的数据存储到指定地址上。<code>LDM</code>和<code>STM</code>指令主要用于现场保护和数据复制。</p><p><code>LDM</code>指令的语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">LDM{addr_mode}{cond}&nbsp;Rn{!},reglist{^}</pre><p><code>STM</code>指令的语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">STM{addr_mode}{cond}&nbsp;Rn{!},reglist{^}</pre><p><code>addr_mode</code>:地址模式,用于数据块传输的地址模式,如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="">IA:每次传送后地址加4。 IB:每次传送前地址加4。 DA:每次传送后地址减4。 DB:每次传送前地址减4。</pre><p>也可以使用相应的面向堆栈的寻址模式,如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="">FD:满递减堆栈(每次传送前地址减4)。 ED:空递增堆栈(每次传送前地址加4)。 FA:满递增堆栈(每次传送后地址减4)。 EA:空递增堆栈(每次传送后地址加4)。</pre><p><code>cond</code>:条件码。</p><p><code>Rn</code>:Rn存储了用于传输的初始地址。</p><p><code>!</code>:如果存在,表示最终的地址要写回<code>Rn</code>。</p><p><code>reglist</code>:用<code>{}</code>括起来的一个寄存器或者多个寄存器组成的列表。它可以是一个寄存器范围。如果<code>{}</code>中的寄存器超过一个,那么寄存器或者寄存器范围之间通过逗号(<code>,</code>)分隔。</p><p><code>^</code>:如果在除了USR模式和SYS模式下存在该符号,意味着将发生下述的两个动作:</p><ul class=" list-paddingleft-2"><li><p>当寄存器列表中不包含PC时,加载/存储的是USR模式的寄存器,而不是当前模式的寄存器。</p></li><li><p>在使用<code>LDM</code>指令时,如果寄存器列表中包含PC时,那么除了正常的多寄存器传送外,会将SPSR 拷贝到CPSR 中,这可用于异常处理返回。</p></li></ul><p>上面的内容可能不是很好理解,下面简单的列举写例子:</p><h3 id="ldmia和stmia例子">LDMIA和STMIA例子</h3><p>LDMIA例子如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;IA:每次传送后地址加4,下面是指令执行流程的分解: #&nbsp;R0=[R1] #&nbsp;R2=[R1+4] #&nbsp;R3=[R1+8] #&nbsp;R4=[R1+12] LDMIA&nbsp;R1,&nbsp;{R0,&nbsp;R2-R4} #&nbsp;IA:每次传送后地址加4,最终地址要写回R1,下面是指令执行流程的分解: #&nbsp;R0=[R1],&nbsp;R1=R1+4 #&nbsp;R2=[R1],&nbsp;R1=R1+4 #&nbsp;R3=[R1],&nbsp;R1=R1+4 #&nbsp;R4=[R1],&nbsp;R1=R1+4 LDMIA&nbsp;R1!,&nbsp;{R0,&nbsp;R2-R4}</pre><p>STMIA例子如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;IA:每次传送后地址加4,下面是指令执行流程的分解: #&nbsp;[R1]=R0 #&nbsp;[R1+4]=R2 #&nbsp;[R1+8]=R3 #&nbsp;[R1+12]=R4 STMIA&nbsp;R1,&nbsp;{R0,&nbsp;R2-R4} #&nbsp;IA:每次传送后地址加4,最终地址要写回R1,下面是指令执行流程的分解: #&nbsp;[R1]=R0,&nbsp;R1=R1+4 #&nbsp;[R1]=R2,&nbsp;R1=R1+4 #&nbsp;[R1]=R3,&nbsp;R1=R1+4 #&nbsp;[R1]=R4,&nbsp;R1=R1+4 STMIA&nbsp;R1!,&nbsp;{R0,&nbsp;R2-R4}</pre><h3 id="ldmib和stmib例子">LDMIB和STMIB例子</h3><p>LDMIB例子如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;IB:每次传送前地址加4,下面是指令执行流程的分解: #&nbsp;R0=[R1+4] #&nbsp;R2=[R1+8] #&nbsp;R3=[R1+12] #&nbsp;R4=[R1+16] LDMIB&nbsp;R1,&nbsp;{R0,&nbsp;R2-R4} #&nbsp;IB:每次传送前地址加4,最终地址要写回R1,下面是指令执行流程的分解: #&nbsp;R1=R1+4,&nbsp;R0=[R1] #&nbsp;R1=R1+4,&nbsp;R2=[R1] #&nbsp;R1=R1+4,&nbsp;R3=[R1] #&nbsp;R1=R1+4,&nbsp;R4=[R1] LDMIB&nbsp;R1!,&nbsp;{R0,&nbsp;R2-R4}</pre><p>STMIB例子如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;IB:每次传送前地址加4,下面是指令执行流程的分解: #&nbsp;[R1+4]=R0 #&nbsp;[R1+8]=R2 #&nbsp;[R1+12]=R3 #&nbsp;[R1+16]=R4 STMIB&nbsp;R1,&nbsp;{R0,&nbsp;R2-R4} #&nbsp;IB:每次传送前地址加4,最终地址要写回R1,下面是指令执行流程的分解: #&nbsp;R1=R1+4,&nbsp;[R1]=R0 #&nbsp;R1=R1+4,&nbsp;[R1]=R2 #&nbsp;R1=R1+4,&nbsp;[R1]=R3 #&nbsp;R1=R1+4,&nbsp;[R1]=R4 STMIB&nbsp;R1!,&nbsp;{R0,&nbsp;R2-R4}</pre><h3 id="ldmda和stmda例子">LDMDA和STMDA例子</h3><p>LDMDA例子如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;DA:每次传送后地址减4,下面是指令执行流程的分解: #&nbsp;R4=[R1] #&nbsp;R3=[R1-4] #&nbsp;R2=[R1-8] #&nbsp;R0=[R1-12] LDMDA&nbsp;R1,&nbsp;{R0,&nbsp;R2-R4} #&nbsp;DA:每次传送后地址减4,最终地址要写回R1,下面是指令执行流程的分解: #&nbsp;R4=[R1],&nbsp;R1=R1-4 #&nbsp;R3=[R1],&nbsp;R1=R1-4 #&nbsp;R2=[R1],&nbsp;R1=R1-4 #&nbsp;R0=[R1],&nbsp;R1=R1-4 LDMDA&nbsp;R1!,&nbsp;{R0,&nbsp;R2-R4}</pre><p>STMDA例子如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;DA:每次传送后地址减4,下面是指令执行流程的分解: #&nbsp;[R1]=R4 #&nbsp;[R1-4]=R3 #&nbsp;[R1-8]=R2 #&nbsp;[R1-12]=R0 STMDA&nbsp;R1,&nbsp;{R0,&nbsp;R2-R4} #&nbsp;DA:每次传送后地址减4,最终地址要写回R1,下面是指令执行流程的分解: #&nbsp;[R1]=R4,&nbsp;R1=R1-4 #&nbsp;[R1]=R3,&nbsp;R1=R1-4 #&nbsp;[R1]=R2,&nbsp;R1=R1-4 #&nbsp;[R1]=R0,&nbsp;R1=R1-4 STMDA&nbsp;R1!,&nbsp;{R0,&nbsp;R2-R4}</pre><h3 id="ldmdb和stmdb例子">LDMDB和STMDB例子</h3><p>LDMDB例子如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;DB:每次传送前地址减4,下面是指令执行流程的分解: #&nbsp;R4=[R1-4] #&nbsp;R3=[R1-8] #&nbsp;R2=[R1-12] #&nbsp;R0=[R1-16] LDMDB&nbsp;R1,&nbsp;{R0,&nbsp;R2-R4} #&nbsp;DB:每次传送前地址减4,最终地址要写回R1,下面是指令执行流程的分解: #&nbsp;R1=R1-4,&nbsp;R4=[R1] #&nbsp;R1=R1-4,&nbsp;R3=[R1] #&nbsp;R1=R1-4,&nbsp;R2=[R1] #&nbsp;R1=R1-4,&nbsp;R0=[R1] LDMDB&nbsp;R1!,&nbsp;{R0,&nbsp;R2-R4}</pre><p>STMDB例子如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;DB:每次传送前地址减4,下面是指令执行流程的分解: #&nbsp;[R1-4]=R4 #&nbsp;[R1-8]=R3 #&nbsp;[R1-12]=R2 #&nbsp;[R1-16]=R0 STMDB&nbsp;R1,&nbsp;{R0,&nbsp;R2-R4} #&nbsp;DB:每次传送前地址减4,最终地址要写回R1,下面是指令执行流程的分解: #&nbsp;R1=R1-4,&nbsp;[R1]=R4 #&nbsp;R1=R1-4,&nbsp;[R1]=R3 #&nbsp;R1=R1-4,&nbsp;[R1]=R2 #&nbsp;R1=R1-4,&nbsp;[R1]=R0 STMDB&nbsp;R1!,&nbsp;{R0,&nbsp;R2-R4}</pre><h3 id="现场保护">现场保护</h3><p>在数据块的传输中:<code>STMDB</code>和<code>LDMIA</code>对应使用,<code>STMIA</code>和<code>LDMDB</code>对应使用。</p><p>在堆栈操作中:<code>STMFD</code>和<code>LDMFD</code>对应使用,<code>STMFA</code>和<code>LDMFA</code>对应使用。</p><p>在子程序或者异常处理时,使用<code>LDMFD</code>和<code>STMFD</code>进行现场保护的例子如下:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;将R0-R7和LR入栈 STMFD&nbsp;SP!,&nbsp;{R0-R7,&nbsp;LR} #&nbsp;功能代码 MOV&nbsp;R0,&nbsp;#0x00 MOV&nbsp;R1,&nbsp;#0x11 MOV&nbsp;R2,&nbsp;#0x22 #&nbsp;将R0-R7和LR出栈 LDMFD&nbsp;SP!,&nbsp;{R0-R7,&nbsp;LR}</pre><p>同样的可以使用<code>STMDB</code>和<code>LDMIA</code>指令进行现场保护,因此上述代码可以修改成下述形式:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;将R0-R7和LR入栈 STMDB&nbsp;SP!,&nbsp;{R0-R7,&nbsp;LR} #&nbsp;功能代码 MOV&nbsp;R0,&nbsp;#0x00 MOV&nbsp;R1,&nbsp;#0x11 MOV&nbsp;R2,&nbsp;#0x22 #&nbsp;将R0-R7和LR出栈 LDMIA&nbsp;SP!,&nbsp;{R0-R7,&nbsp;LR}</pre><h2 id="push和pop指令">push和pop指令</h2><p><code>push</code>和<code>pop</code>指令主要用于子程序或者异常的现场保护。<code>push</code>指令用于将寄存器内容压入堆栈。<code>pop</code>指令用于将堆栈中的内容恢复到寄存器中。</p><p><code>push</code>指令的语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">PUSH{cond}&nbsp;reglist</pre><p><code>pop</code>指令的语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">POP{cond}&nbsp;reglist</pre><p><code>cond</code>:条件码。</p><p><code>reglist</code>:用<code>{}</code>括起来的一个寄存器或者多个寄存器组成的列表。它可以是一个寄存器范围。如果<code>{}</code>中的寄存器超过一个,那么寄存器或者寄存器范围之间通过逗号(<code>,</code>)分隔。</p><p><code>push</code>指令等价于<code>STMDB</code>指令。<code>pop</code>指令等价于<code>LDMIA</code>指令。</p><p>使用<code>push</code>指令和<code>pop</code>指令保护现场的例子如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;将R0-R7和LR入栈 push&nbsp;{R0-R7,&nbsp;LR} #&nbsp;功能代码 MOV&nbsp;R0,&nbsp;#0x00 MOV&nbsp;R1,&nbsp;#0x11 MOV&nbsp;R2,&nbsp;#0x22 #&nbsp;将R0-R7和LR出栈 pop&nbsp;{R0-R7,&nbsp;LR}</pre><h2 id="mov指令">MOV指令</h2><p><code>MOV</code>指令主要用于将数据搬移到寄存器中。<code>MOV</code>指令的语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">MOV{S}{cond}&nbsp;Rn,&nbsp;Rm MOV{cond}&nbsp;Rn,&nbsp;#imm</pre><p><code>S</code>:可选的后缀,如果指令中添加了<code>S</code>,那么指令的执行结果将会影响到CPSR寄存器的标志位域。</p><p><code>cond</code>:条件码。</p><p><code>Rn</code>:目标寄存器。</p><p><code>Rm</code>:源寄存器。</p><p><code>imm</code>:立即数。</p><p><code>MOV</code>指令的使用例子如下:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;将&nbsp;R1&nbsp;寄存器中的内容搬移到&nbsp;R0&nbsp;寄存器 MOV&nbsp;R0,&nbsp;R1 #&nbsp;将&nbsp;0xaa&nbsp;搬移到&nbsp;R0&nbsp;寄存器 MOV&nbsp;R0,&nbsp;#0xaa</pre><h2 id="cps指令">CPS指令</h2><p>可以通过<code>CPS</code>(Change Processor State)指令来修改处理器模式。<code>CPS</code>指令也可以用来使能或者禁止异常。</p><p><code>CPS</code>指令的语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">CPS&nbsp;#mode CPSIE&nbsp;iflags{,&nbsp;#mode} CPSID&nbsp;iflags{,&nbsp;#mode}</pre><p><code>mode</code>是处理器的模式编码,比如在从其他模式下切换到SYS模式,使用下述代码即可:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;切换到SYS模式 CPS&nbsp;#0x1f</pre><p><code>IE</code>使能中断或者终止。</p><p><code>ID</code>禁止中断或者终止。</p><p><code>iflags</code>由下面的一种或者几种组成:</p><ul class=" list-paddingleft-2"><li><p><code>a</code>:表示异步终止(asynchronous abort);</p></li><li><p><code>i</code>:表示中断(IRQ);</p></li><li><p><code>f</code>:表示快中断(FIQ);</p></li></ul><p>下述代码是<code>CPS</code>指令的一些简单用法:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm" style="break-inside: unset;">#&nbsp;使能中断 CPSIE&nbsp;I #&nbsp;禁止中断 CPSID&nbsp;I #&nbsp;使能异步终止和快中断 CPSIE&nbsp;AF #&nbsp;禁止异步终止和快中断 CPSID&nbsp;AF #&nbsp;使能中断并切换到SYS模式 CPSIE&nbsp;I,&nbsp;#0x1f</pre><h2 id="mrs与msr指令">MRS与MSR指令</h2><p><code>MRS</code>和<code>MSR</code>指令可用于读写程序状态寄存器CPSR,APSR和SPSR。</p><p>在ARM处理器中,只有<code>MRS</code>指令可以从程序状态寄存器CPSR,APSR和SPSR中读出数据到通用寄存器中。<code>MRS</code>指令操作程序状态寄存器的语法如下:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">MRS{cond}&nbsp;Rd,&nbsp;psr</pre><p><code>cond</code>为条件码。</p><p><code>Rd</code>为目标寄存器,<code>Rd</code>不允许为R15。</p><p><code>psr</code>为程序状态寄存器CPSR,APSR或者SPSR。</p><p><code>MRS</code>指令的示例代码如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;将CPSR寄存器的值读取到R0中 MRS&nbsp;R0,&nbsp;CPSR #&nbsp;将SPSR寄存器的值读取到R1中 MRS&nbsp;R1,&nbsp;SPSR #&nbsp;将APSR寄存器的值读取到R2中 MRS&nbsp;R2,&nbsp;APSR</pre><p><code>MSR</code>指令可以用来写程序状态寄存器CPSR,APSR和SPSR的全部或者部分域。<code>MSR</code>指令操作程序状态寄存器的语法如下:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">MSR{cond}&nbsp;psr,&nbsp;#constant MSR{cond}&nbsp;psr,&nbsp;Rm MSR{cond}&nbsp;psr_fields,&nbsp;#constant MSR{cond}&nbsp;psr_fields,&nbsp;Rm</pre><p><code>cond</code>为条件码。</p><p><code>psr</code>为程序状态寄存器CPSR或者SPSR。</p><p><code>constant</code>是一个8位立即数。ARM文档对于<code>constant</code>的介绍如下:</p><blockquote><p>constant is an 8-bit pattern rotated by an even number of bits within a 32-bit word. (Not available in Thumb.)</p></blockquote><p><code>Rm</code>是源寄存器。</p><p><code>fields</code>由下面的一个或者多个组合而成:</p><ul class=" list-paddingleft-2"><li><p><code>c</code>:xPSR[7:0],控制位域;</p></li><li><p><code>x</code>:xPSR[15:8],扩展位域;</p></li><li><p><code>s</code>:xPSR[23:16],状态位域;</p></li><li><p><code>f</code>:xPSR[31:24],标志位域;</p></li></ul><p><code>MSR</code>指令的示例代码如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;切换到SYS模式 MRS&nbsp;R0,&nbsp;CPSR ORR&nbsp;R0,&nbsp;R0,&nbsp;#0x1f MSR&nbsp;CPSR,&nbsp;R0 #&nbsp;切换到SYS模式 MSR&nbsp;CPSR_c,&nbsp;#0xDF</pre><p><strong>只有在除用户模式外的其他模式下才能够修改状态寄存器</strong>。</p><h2 id="mrc和mcr指令">MRC和MCR指令</h2><p>ARMv7-A体系结构的处理器提供了<code>MRC</code>和<code>MCR</code>指令用于对协处理器进行读写操作。<code>MRC</code>指令用于将协处理器中的寄存器数据读取到ARM通用寄存器中。<code>MCR</code>指令用于将ARM通用寄存器中的数据写入到协处理器的寄存器中。</p><h3 id="mrc">MRC</h3><p><code>MRC</code>指令的语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">MRC{cond}&nbsp;coproc,&nbsp;opc1,&nbsp;Rt,&nbsp;CRn,&nbsp;CRm{,&nbsp;opc2}</pre><p><code>cond</code>为条件码。</p><p><code>coproc</code>为协处理器名称,CP0~CP15协处理器分别对应名称p0~p15。</p><p><code>opc1</code>为协处理器要执行的操作码,取指范围为0~7。</p><p><code>Rt</code>为ARM通用寄存器,用于存储读取到的协处理器寄存器数据。</p><p><code>CRn</code>为协处理器寄存器,对于CP15协处理器来说,<code>CRn</code>取值范围为c0~c15。</p><p><code>CRm</code>为协处理器寄存器,对于CP15协处理器来说,通过<code>CRm</code>和<code>opc2</code>一起来确定<code>CRn</code>对应的具体寄存器。</p><p><code>opc2</code>为可选的协处理器执行操作码,取指范围为0~7,当不需要的时候要设置为0。</p><p><code>MRC</code>指令使用示例如下:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;读取主ID寄存器&nbsp;MIDR&nbsp;的数据到&nbsp;R0&nbsp;中. MRC&nbsp;p15,&nbsp;0,&nbsp;R0,&nbsp;c0,&nbsp;c0,&nbsp;0</pre><h3 id="mcr">MCR</h3><p><code>MCR</code>指令的语法如下所示:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">MCR{cond}&nbsp;coproc,&nbsp;opc1,&nbsp;Rt,&nbsp;CRn,&nbsp;CRm{,&nbsp;opc2}</pre><p><code>cond</code>为条件码。</p><p><code>coproc</code>为协处理器名称,CP0~CP15协处理器分别对应名称p0~p15。</p><p><code>opc1</code>为协处理器要执行的操作码,取指范围为0~7。</p><p><code>Rt</code>为ARM通用寄存器,用于存储要写入到协处理器寄存器中的数据。</p><p><code>CRn</code>为协处理器寄存器,对于CP15协处理器来说,<code>CRn</code>取值范围为c0~c15。</p><p><code>CRm</code>为协处理器寄存器,对于CP15协处理器来说,通过<code>CRm</code>和<code>opc2</code>一起来确定<code>CRn</code>对应的具体寄存器。</p><p><code>opc2</code>为可选的协处理器执行操作码,取指范围为0~7,当不需要的时候要设置为0。</p><p><code>MCR</code>指令使用示例如下:</p><pre class="md-fences md-end-block ty-contain-cm modeLoaded" spellcheck="false" lang="asm">#&nbsp;将&nbsp;R0&nbsp;中的配置数据写入到&nbsp;SCTLR MCR&nbsp;p15,&nbsp;0,&nbsp;R0,&nbsp;c1,&nbsp;c0,&nbsp;0</pre><!--!doctype-->
欢迎关注博主的公众号呀,精彩内容随时掌握:
热情邀请仔细浏览下博客中的广告,万一有对自己有用或感兴趣的呢。◕ᴗ◕。。
如果这篇文章对你有帮助,记得点赞和关注博主就行了^_^,当然了能够赞赏博主,那就非常感谢啦!
注: 转载请注明出处,谢谢!^_^
转载请注明来源: 本文链接:  By: AilsonJack
ARMv7-A 那些事 - 6.常用汇编指令  |  说好一起走
暂无评论,要不要来个沙发
发表评论

 
Copyright © 2015~2023  说好一起走   保留所有权利   |  百度统计  蜀ICP备15004292号