arithmetic shift

{{short description|Shift operator in computer programming}}

Image:Rotate right arithmetically.svg

Image:Rotate left logically.svg is filled with a zero.]]

class="wikitable" style="float:right; clear:right;"

|+ Arithmetic shift operators in various programming languages and processors

! Language or processor !! Left !! Right

style="max-width:195px" | ActionScript 3, Java, JavaScript, Python, PHP, Ruby, C, C++,{{Cite web|url=https://tour.dlang.org/tour/en/gems/bit-manipulation|title=Bit manipulation - Dlang Tour|website=tour.dlang.org|access-date=2019-06-23}}D, C#, Go, Julia,
Rust (signed types only){{Cite web|title=Operator Expressions: Arithmetic and Logical Binary Operators|url=https://doc.rust-lang.org/reference/expressions/operator-expr.html#arithmetic-and-logical-binary-operators|access-date=2022-11-13|website=doc.rust-lang.org}},
Swift (signed types only)The >> operator in C and C++ is not necessarily an arithmetic shift. Usually it is only an arithmetic shift if used with a signed integer type on its left-hand side. If it is used on an unsigned integer type instead, it will be a logical shift.
<< >>
Ada Shift_Left {{Cite web|url=https://www.adaic.org/resources/add_content/standards/12aarm/html/AA-B-2.html|title=Annotated Ada 2012 Reference Manual}} Shift_Right_Arithmetic
Kotlin shl shr
Fortran SHIFTL SHIFTA Fortran 2008.
Standard ML << ~>>
Verilog <<< >>> The Verilog arithmetic right shift operator only actually performs an arithmetic shift if the first operand is signed. If the first operand is unsigned, the operator actually performs a logical right shift.
OpenVMS macro language

| colspan="2" align="center" | @{{#tag:ref|In the OpenVMS macro language, whether an arithmetic shift is left or right is determined by whether the second operand is positive or negative. This is unusual. In most programming languages the two directions have distinct operators, with the operator specifying the direction, and the second operand is implicitly positive. (Some languages, such as Verilog, require that negative values be converted to unsigned positive values. Some languages, such as C and C++, have no defined behaviour if negative values are used.){{sfn|HP|2001}}{{Page needed|date=March 2012}}|group="note"}}

Scheme

| colspan="2" align="center" | arithmetic-shiftIn Scheme arithmetic-shift can be both left and right shift, depending on the second operand, very similar to the OpenVMS macro language, although R6RS Scheme adds both -right and -left variants.

Common Lisp

| colspan="2" align="center" | ash

OCamllslasr
Haskell

| colspan="2" align="center" | Data.Bits.shift{{#tag:ref|The Bits class from Haskell's Data.Bits module defines both shift taking a signed argument and shiftL/shiftR taking unsigned arguments. These are isomorphic; for new definitions the programmer need provide only one of the two forms and the other form will be automatically defined in terms of the provided one.|group="note"}}

VHDLslaThe VHDL arithmetic left shift operator is unusual. Instead of filling the LSB of the result with zero, it copies the original LSB into the new LSB. While this is an exact mirror image of the arithmetic right shift, it is not the conventional definition of the operator, and is not equivalent to multiplication by a power of 2. In the VHDL 2008 standard this strange behavior was left unchanged (for backward compatibility) for argument types that do not have forced numeric interpretation (e.g., BIT_VECTOR) but 'SLA' for unsigned and signed argument types behaves in the expected way (i.e., rightmost positions are filled with zeros). VHDL's shift left logical (SLL) function does implement the aforementioned 'standard' arithmetic shift.sra
Assembly: Z80SLA{{Cite web|url=http://www.z80.info/z80syntx.htm#SLA|title=Z80 Assembler Syntax}}SRA
Assembly: x86SALSAR
Assembly: 68kASLASR
Assembly: RISC-Vsll, slli{{Cite web|url=https://github.com/riscv/riscv-isa-manual/releases/download/Ratified-IMAFDQC/riscv-spec-20191213.pdf |archive-url=https://ghostarchive.org/archive/20221009/https://github.com/riscv/riscv-isa-manual/releases/download/Ratified-IMAFDQC/riscv-spec-20191213.pdf |archive-date=2022-10-09 |url-status=live|date=2019-12-13|access-date=2021-08-07|title=The RISC-V Instruction Set Manual, Volume I: Unprivileged ISA|website=GitHub |pages=18–20}}sra, srai

In computer programming, an arithmetic shift is a shift operator, sometimes termed a signed shift (though it is not restricted to signed operands). The two basic types are the arithmetic left shift and the arithmetic right shift. For binary numbers it is a bitwise operation that shifts all of the bits of its operand; every bit in the operand is simply moved a given number of bit positions, and the vacant bit-positions are filled in. Instead of being filled with all 0s, as in logical shift, when shifting to the right, the leftmost bit (usually the sign bit in signed integer representations) is replicated to fill in all the vacant positions (this is a kind of sign extension).

Some authors prefer the terms sticky right-shift and zero-fill right-shift for arithmetic and logical shifts respectively.

Thomas R. Cain and Alan T. Sherman.

[https://web.archive.org/web/20140109191811/http://www.cisa.umbc.edu/papers/ShermanCryptologia97.pdf "How to break Gifford's cipher"].

Section 8.1: "Sticky versus Non-Sticky Bit-shifting".

Cryptologia.

1997.

Arithmetic shifts can be useful as efficient ways to perform multiplication or division of signed integers by powers of two. Shifting left by n bits on a signed or unsigned binary number has the effect of multiplying it by 2n. Shifting right by n bits on a two's complement signed binary number has the effect of dividing it by 2n, but it always rounds down (towards negative infinity). This is different from the way rounding is usually done in signed integer division (which rounds towards 0). This discrepancy has led to bugs in a number of compilers.{{cite web|last=Steele Jr|first=Guy|title=Arithmetic Shifting Considered Harmful|url=http://dspace.mit.edu/bitstream/handle/1721.1/6090/AIM-378.pdf |archive-url=https://ghostarchive.org/archive/20221009/http://dspace.mit.edu/bitstream/handle/1721.1/6090/AIM-378.pdf |archive-date=2022-10-09 |url-status=live|publisher=MIT AI Lab|access-date=20 May 2013}}

For example, in the x86 instruction set, the SAR instruction (arithmetic right shift) divides a signed number by a power of two, rounding towards negative infinity.{{sfn|Hyde|1996|loc=§ 6.6.2.2 SAR}} However, the IDIV instruction (signed divide) divides a signed number, rounding towards zero. So a SAR instruction cannot be substituted for an IDIV by power of two instruction nor vice versa.

Formal definition

The formal definition of an arithmetic shift, from Federal Standard 1037C is that it is:

{{bq|text=A shift, applied to the representation of a number in a fixed radix numeration system and in a fixed-point representation system, and in which only the characters representing the fixed-point part of the number are moved. An arithmetic shift is usually equivalent to multiplying the number by a positive or a negative integral power of the radix, except for the effect of any rounding; compare the logical shift with the arithmetic shift, especially in the case of floating-point representation.}}

An important word in the FS 1073C definition is "usually".

= Non-equivalence of arithmetic right shift and division =

However, arithmetic right shifts are major traps for the unwary, specifically in treating rounding of negative integers. For example, in the usual two's complement representation of negative integers, −1 is represented as all 1's. For an 8-bit signed integer this is 1111 1111. An arithmetic right-shift by 1 (or 2, 3, ..., 7) yields 1111 1111 again, which is still −1. This corresponds to rounding down (towards negative infinity), but is not the usual convention for division.

It is frequently stated that arithmetic right shifts are equivalent to division by a (positive, integral) power of the radix (e.g., a division by a power of 2 for binary numbers), and hence that division by a power of the radix can be optimized by implementing it as an arithmetic right shift. (A shifter is much simpler than a divider. On most processors, shift instructions will execute faster than division instructions.) Large number of 1960s and 1970s programming handbooks, manuals, and other specifications from companies and institutions such as DEC, IBM, Data General, and ANSI make such incorrect statements{{sfn|Steele|1977|p=}}{{Page needed|date=March 2012}}.

Logical right shifts are equivalent to division by a power of the radix (usually 2) only for positive or unsigned numbers. Arithmetic right shifts are equivalent to logical right shifts for positive signed numbers. Arithmetic right shifts for negative numbers in N's complement (usually two's complement) is roughly equivalent to division by a power of the radix (usually 2), where for odd numbers rounding downwards is applied (not towards 0 as usually expected).

Arithmetic right shifts for negative numbers are equivalent to division using rounding towards 0 in ones' complement representation of signed numbers as was used by some historic computers, but this is no longer in general use.

== Handling the issue in programming languages ==

The (1999) ISO standard for the programming language C defines the right shift operator in terms of divisions by powers of 2.{{sfn|ISOIEC9899|1999|loc=§ 6.5.7 Bitwise shift operators}} Because of the above-stated non-equivalence, the standard explicitly excludes from that definition the right shifts of signed numbers that have negative values. It does not specify the behaviour of the right shift operator in such circumstances, but instead requires each individual C compiler to define the behaviour of shifting negative values right.{{#tag:ref|The C standard was intended to not restrict the C language to either ones' complement or two's complement architectures. In cases where the behaviours of ones' complement and two's complement representations differ, such as this, the standard requires individual C compilers to document the behaviour of their target architectures. The documentation for GNU Compiler Collection (GCC), for example, documents its behaviour as employing sign-extension.{{sfn|FSF|2008|loc=§ 4.5 Integers implementation}}|group="note"}}

Like C, C++ had an implementation-defined right shift for signed integers until C++20. Starting in the C++20 standard, right shift of a signed integer is defined to be an arithmetic shift.{{sfn|ISOCPP20|2020|loc=§ 7.6.7 Shift operators}}

Applications

In applications where consistent rounding down is desired, arithmetic right shifts for signed values are useful. An example is in downscaling raster coordinates by a power of two, which maintains even spacing. For example, right shift by 1 sends 0, 1, 2, 3, 4, 5, ... to 0, 0, 1, 1, 2, 2, ..., and −1, −2, −3, −4, ... to −1, −1, −2, −2, ..., maintaining even spacing as −2, −2, −1, −1, 0, 0, 1, 1, 2, 2, ... In contrast, integer division with rounding towards zero sends −1, 0, and 1 all to 0 (3 points instead of 2), yielding −2, −1, −1, 0, 0, 0, 1, 1, 2, 2, ... instead, which is irregular at 0.

Notes

References

= Cross-reference =

{{Reflist|30em}}

= Sources used =

{{FS1037C}}

{{Refbegin}}

  • {{cite book|first=Donald|last=Knuth|title=The Art of Computer Programming, Volume 2 — Seminumerical algorithms|pages=169–170|location=Reading, Mass.|publisher=Addison-Wesley|year=1969|author-link=Donald Knuth}}
  • {{cite journal|title=Arithmetic shifting considered harmful|journal=ACM SIGPLAN Notices Archive|volume=12|issue=11|date=November 1977|pages=61–69|first=Guy L.|last=Steele|publisher=ACM Press|location=New York|doi=10.1145/956641.956647|url=http://www.dtic.mil/get-tr-doc/pdf?AD=ADA031883|archive-url=https://web.archive.org/web/20170922201035/http://www.dtic.mil/get-tr-doc/pdf?AD=ADA031883|url-status=dead|archive-date=September 22, 2017|hdl=1721.1/6090|s2cid=15420308 |hdl-access=free}}
  • {{cite book |ref=CITEREFHP2001|chapter-url=http://h71000.www7.hp.com/doc/73FINAL/4515/4515pro_002.html#8_arithmeticshiftoperator |work=HP OpenVMS Systems Documentation|title=VAX MACRO and Instruction Set Reference Manual |chapter=3.7.1 Arithmetic Shift Operator|publisher=Hewlett-Packard Development Company|date=April 2001 |archive-url=https://web.archive.org/web/20110808085326/http://h71000.www7.hp.com/doc/73final/4515/4515pro_002.html#8_arithmeticshiftoperator |archive-date=2011-08-08}}
  • {{Cite book |ref=CITEREFISOIEC98991999|title=Programming languages — C|publisher=International Organization for Standardization|year=1999|version=ISO/IEC 9899:1999}}
  • {{cite book|chapter=CHAPTER SIX: THE 80x86 INSTRUCTION SET (Part 3)|date=1996-09-26|title=The Art of ASSEMBLY LANGUAGE PROGRAMMING|first=Randall|last=Hyde|url=http://www.arl.wustl.edu/~lockwood/class/cs306/books/artofasm/Chapter_6/CH06-3.html#HEADING3-120|access-date=2007-11-28|archive-url=https://web.archive.org/web/20071123223102/http://www.arl.wustl.edu/~lockwood/class/cs306/books/artofasm/Chapter_6/CH06-3.html#HEADING3-120|archive-date=2007-11-23|url-status=dead}}
  • {{cite web|ref=CITEREFFSF2008|year=2008|publisher=Free Software Foundation|url=https://gcc.gnu.org/onlinedocs/gcc-4.3.3/gcc/Integers-implementation.html#Integers-implementation|work=GCC manual|title=C Implementation}}
  • {{cite web|ref={{harvid|ISOCPP20|2020}} |year=2020|publisher=International Organization for Standardization|title = ISO/IEC 14882:2020(E) – Programming Language C++|url = https://isocpp.org/files/papers/N4860.pdf}}

{{refend}}

{{DEFAULTSORT:Arithmetic Shift}}

Category:Binary arithmetic

Category:Operators (programming)