]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - contrib/llvm-project/llvm/lib/Target/AVR/AVRInstrInfo.td
Merge llvm, clang, compiler-rt, libc++, libunwind, lld, lldb and openmp
[FreeBSD/FreeBSD.git] / contrib / llvm-project / llvm / lib / Target / AVR / AVRInstrInfo.td
1 //===-- AVRInstrInfo.td - AVR Instruction defs -------------*- tablegen -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file describes the AVR instructions in TableGen format.
10 //
11 //===----------------------------------------------------------------------===//
12
13 include "AVRInstrFormats.td"
14
15 //===----------------------------------------------------------------------===//
16 // AVR Type Profiles
17 //===----------------------------------------------------------------------===//
18
19 def SDT_AVRCallSeqStart : SDCallSeqStart<[SDTCisVT<0, i16>, SDTCisVT<1, i16>]>;
20 def SDT_AVRCallSeqEnd : SDCallSeqEnd<[SDTCisVT<0, i16>, SDTCisVT<1, i16>]>;
21 def SDT_AVRCall : SDTypeProfile<0, -1, [SDTCisVT<0, iPTR>]>;
22 def SDT_AVRWrapper : SDTypeProfile<1, 1, [SDTCisSameAs<0, 1>, SDTCisPtrTy<0>]>;
23 def SDT_AVRBrcond : SDTypeProfile<0, 2,
24                                   [SDTCisVT<0, OtherVT>, SDTCisVT<1, i8>]>;
25 def SDT_AVRCmp : SDTypeProfile<0, 2, [SDTCisSameAs<0, 1>]>;
26 def SDT_AVRTst : SDTypeProfile<0, 1, [SDTCisInt<0>]>;
27 def SDT_AVRSelectCC : SDTypeProfile<1, 3, [SDTCisSameAs<0, 1>,
28                                     SDTCisSameAs<1, 2>, SDTCisVT<3, i8>]>;
29
30 //===----------------------------------------------------------------------===//
31 // AVR Specific Node Definitions
32 //===----------------------------------------------------------------------===//
33
34 def AVRretflag : SDNode<"AVRISD::RET_FLAG", SDTNone,
35                         [SDNPHasChain, SDNPOptInGlue, SDNPVariadic]>;
36 def AVRretiflag : SDNode<"AVRISD::RETI_FLAG", SDTNone,
37                          [SDNPHasChain, SDNPOptInGlue, SDNPVariadic]>;
38
39 def AVRcallseq_start : SDNode<"ISD::CALLSEQ_START", SDT_AVRCallSeqStart,
40                               [SDNPHasChain, SDNPOutGlue]>;
41 def AVRcallseq_end : SDNode<"ISD::CALLSEQ_END", SDT_AVRCallSeqEnd,
42                             [SDNPHasChain, SDNPOptInGlue, SDNPOutGlue]>;
43
44 def AVRcall : SDNode<"AVRISD::CALL", SDT_AVRCall,
45                      [SDNPHasChain, SDNPOutGlue, SDNPOptInGlue, SDNPVariadic]>;
46
47 def AVRWrapper : SDNode<"AVRISD::WRAPPER", SDT_AVRWrapper>;
48
49 def AVRbrcond : SDNode<"AVRISD::BRCOND", SDT_AVRBrcond,
50                        [SDNPHasChain, SDNPInGlue]>;
51 def AVRcmp : SDNode<"AVRISD::CMP", SDT_AVRCmp, [SDNPOutGlue]>;
52 def AVRcmpc : SDNode<"AVRISD::CMPC", SDT_AVRCmp, [SDNPInGlue, SDNPOutGlue]>;
53 def AVRtst : SDNode<"AVRISD::TST", SDT_AVRTst, [SDNPOutGlue]>;
54 def AVRselectcc: SDNode<"AVRISD::SELECT_CC", SDT_AVRSelectCC, [SDNPInGlue]>;
55
56 // Shift nodes.
57 def AVRlsl : SDNode<"AVRISD::LSL", SDTIntUnaryOp>;
58 def AVRlsr : SDNode<"AVRISD::LSR", SDTIntUnaryOp>;
59 def AVRrol : SDNode<"AVRISD::ROL", SDTIntUnaryOp>;
60 def AVRror : SDNode<"AVRISD::ROR", SDTIntUnaryOp>;
61 def AVRasr : SDNode<"AVRISD::ASR", SDTIntUnaryOp>;
62
63 // Pseudo shift nodes for non-constant shift amounts.
64 def AVRlslLoop : SDNode<"AVRISD::LSLLOOP", SDTIntShiftOp>;
65 def AVRlsrLoop : SDNode<"AVRISD::LSRLOOP", SDTIntShiftOp>;
66 def AVRrolLoop : SDNode<"AVRISD::ROLLOOP", SDTIntShiftOp>;
67 def AVRrorLoop : SDNode<"AVRISD::RORLOOP", SDTIntShiftOp>;
68 def AVRasrLoop : SDNode<"AVRISD::ASRLOOP", SDTIntShiftOp>;
69
70 //===----------------------------------------------------------------------===//
71 // AVR Operands, Complex Patterns and Transformations Definitions.
72 //===----------------------------------------------------------------------===//
73
74 def imm8_neg_XFORM : SDNodeXForm<imm,
75 [{
76   return CurDAG->getTargetConstant(-N->getAPIntValue(), SDLoc(N), MVT::i8);
77 }]>;
78
79 def imm16_neg_XFORM : SDNodeXForm<imm,
80 [{
81   return CurDAG->getTargetConstant(-N->getAPIntValue(), SDLoc(N), MVT::i16);
82 }]>;
83
84 def imm0_63_neg : PatLeaf<(imm),
85 [{
86   int64_t val = -N->getSExtValue();
87   return val >= 0 && val < 64;
88 }], imm16_neg_XFORM>;
89
90 def uimm6 : PatLeaf<(imm), [{ return isUInt<6>(N->getZExtValue()); }]>;
91
92 // imm_com8_XFORM - Return the complement of a imm_com8 value
93 def imm_com8_XFORM : SDNodeXForm<imm, [{
94   return CurDAG->getTargetConstant(~((uint8_t)N->getZExtValue()), SDLoc(N),
95                                    MVT::i8);
96 }]>;
97
98 // imm_com8 - Match an immediate that is a complement
99 // of a 8-bit immediate.
100 // Note: this pattern doesn't require an encoder method and such, as it's
101 // only used on aliases (Pat<> and InstAlias<>). The actual encoding
102 // is handled by the destination instructions, which use imm_com8.
103 def imm_com8_asmoperand : AsmOperandClass { let Name = "ImmCom8"; }
104 def imm_com8 : Operand<i8> {
105   let ParserMatchClass = imm_com8_asmoperand;
106 }
107
108 def ioaddr_XFORM : SDNodeXForm<imm,
109 [{
110   return CurDAG->getTargetConstant(uint8_t(N->getZExtValue()) - 0x20, SDLoc(N), MVT::i8);
111 }]>;
112
113 def iobitpos8_XFORM : SDNodeXForm<imm,
114 [{
115   return CurDAG->getTargetConstant(Log2_32(uint8_t(N->getZExtValue())),
116                                    SDLoc(N), MVT::i8);
117 }]>;
118
119 def iobitposn8_XFORM : SDNodeXForm<imm,
120 [{
121   return CurDAG->getTargetConstant(Log2_32(uint8_t(~N->getZExtValue())),
122                                    SDLoc(N), MVT::i8);
123 }]>;
124
125 def ioaddr8 : PatLeaf<(imm),
126 [{
127   uint64_t val = N->getZExtValue();
128   return val >= 0x20 && val < 0x60;
129 }], ioaddr_XFORM>;
130
131 def lowioaddr8 : PatLeaf<(imm),
132 [{
133   uint64_t val = N->getZExtValue();
134   return val >= 0x20 && val < 0x40;
135 }], ioaddr_XFORM>;
136
137 def ioaddr16 : PatLeaf<(imm),
138 [{
139   uint64_t val = N->getZExtValue();
140   return val >= 0x20 && val < 0x5f;
141 }], ioaddr_XFORM>;
142
143 def iobitpos8 : PatLeaf<(imm),
144 [{
145   return isPowerOf2_32(uint8_t(N->getZExtValue()));
146 }], iobitpos8_XFORM>;
147
148 def iobitposn8 : PatLeaf<(imm),
149 [{
150   return isPowerOf2_32(uint8_t(~N->getZExtValue()));
151 }], iobitposn8_XFORM>;
152
153 def MemriAsmOperand : AsmOperandClass {
154   let Name = "Memri";
155   let ParserMethod = "parseMemriOperand";
156 }
157
158 /// Address operand for `reg+imm` used by STD and LDD.
159 def memri : Operand<iPTR>
160 {
161   let MIOperandInfo = (ops PTRDISPREGS, i16imm);
162
163   let PrintMethod = "printMemri";
164   let EncoderMethod = "encodeMemri";
165
166   let ParserMatchClass = MemriAsmOperand;
167 }
168
169 // Address operand for `SP+imm` used by STD{W}SPQRr
170 def memspi : Operand<iPTR>
171 {
172   let MIOperandInfo = (ops GPRSP, i16imm);
173 }
174
175 def relbrtarget_7 : Operand<OtherVT>
176 {
177     let PrintMethod   = "printPCRelImm";
178     let EncoderMethod = "encodeRelCondBrTarget<AVR::fixup_7_pcrel>";
179 }
180
181 def brtarget_13 : Operand<OtherVT>
182 {
183     let PrintMethod   = "printPCRelImm";
184     let EncoderMethod = "encodeRelCondBrTarget<AVR::fixup_13_pcrel>";
185 }
186
187 // The target of a 22 or 16-bit call/jmp instruction.
188 def call_target : Operand<iPTR>
189 {
190     let EncoderMethod = "encodeCallTarget";
191 }
192
193 // A 16-bit address (which can lead to an R_AVR_16 relocation).
194 def imm16 : Operand<i16>
195 {
196     let EncoderMethod = "encodeImm<AVR::fixup_16, 2>";
197 }
198
199 /// A 6-bit immediate used in the ADIW/SBIW instructions.
200 def imm_arith6 : Operand<i16>
201 {
202     let EncoderMethod = "encodeImm<AVR::fixup_6_adiw, 0>";
203 }
204
205 /// An 8-bit immediate inside an instruction with the same format
206 /// as the `LDI` instruction (the `FRdK` format).
207 def imm_ldi8 : Operand<i8>
208 {
209     let EncoderMethod = "encodeImm<AVR::fixup_ldi, 0>";
210 }
211
212 /// A 5-bit port number used in SBIC and friends (the `FIOBIT` format).
213 def imm_port5 : Operand<i8>
214 {
215     let EncoderMethod = "encodeImm<AVR::fixup_port5, 0>";
216 }
217
218 /// A 6-bit port number used in the `IN` instruction and friends (the
219 /// `FIORdA` format.
220 def imm_port6 : Operand<i8>
221 {
222     let EncoderMethod = "encodeImm<AVR::fixup_port6, 0>";
223 }
224
225 // Addressing mode pattern reg+imm6
226 def addr : ComplexPattern<iPTR, 2, "SelectAddr", [], [SDNPWantRoot]>;
227
228 // AsmOperand class for a pointer register.
229 // Used with the LD/ST family of instructions.
230 // See FSTLD in AVRInstrFormats.td
231 def PtrRegAsmOperand : AsmOperandClass
232 {
233    let Name = "Reg";
234 }
235
236 // A special operand type for the LD/ST instructions.
237 // It converts the pointer register number into a two-bit field used in the
238 // instruction.
239 def LDSTPtrReg : Operand<i16>
240 {
241     let MIOperandInfo = (ops PTRREGS);
242     let EncoderMethod = "encodeLDSTPtrReg";
243
244     let ParserMatchClass = PtrRegAsmOperand;
245 }
246
247 // A special operand type for the LDD/STD instructions.
248 // It behaves identically to the LD/ST version, except restricts
249 // the pointer registers to Y and Z.
250 def LDDSTDPtrReg : Operand<i16>
251 {
252     let MIOperandInfo = (ops PTRDISPREGS);
253     let EncoderMethod = "encodeLDSTPtrReg";
254
255     let ParserMatchClass = PtrRegAsmOperand;
256 }
257
258 //===----------------------------------------------------------------------===//
259 // AVR predicates for subtarget features
260 //===----------------------------------------------------------------------===//
261
262 def HasSRAM       :    Predicate<"Subtarget->hasSRAM()">,
263                          AssemblerPredicate<"FeatureSRAM">;
264
265 def HasJMPCALL    :    Predicate<"Subtarget->hasJMPCALL()">,
266                          AssemblerPredicate<"FeatureJMPCALL">;
267
268 def HasIJMPCALL   :    Predicate<"Subtarget->hasIJMPCALL()">,
269                          AssemblerPredicate<"FeatureIJMPCALL">;
270
271 def HasEIJMPCALL  :    Predicate<"Subtarget->hasEIJMPCALL()">,
272                          AssemblerPredicate<"FeatureEIJMPCALL">;
273
274 def HasADDSUBIW   :    Predicate<"Subtarget->hasADDSUBIW()">,
275                          AssemblerPredicate<"FeatureADDSUBIW">;
276
277 def HasSmallStack :    Predicate<"Subtarget->HasSmallStack()">,
278                          AssemblerPredicate<"FeatureSmallStack">;
279
280 def HasMOVW       :    Predicate<"Subtarget->hasMOVW()">,
281                          AssemblerPredicate<"FeatureMOVW">;
282
283 def HasLPM        :    Predicate<"Subtarget->hasLPM()">,
284                          AssemblerPredicate<"FeatureLPM">;
285
286 def HasLPMX       :    Predicate<"Subtarget->hasLPMX()">,
287                          AssemblerPredicate<"FeatureLPMX">;
288
289 def HasELPM       :    Predicate<"Subtarget->hasELPM()">,
290                          AssemblerPredicate<"FeatureELPM">;
291
292 def HasELPMX      :    Predicate<"Subtarget->hasELPMX()">,
293                          AssemblerPredicate<"FeatureELPMX">;
294
295 def HasSPM        :    Predicate<"Subtarget->hasSPM()">,
296                          AssemblerPredicate<"FeatureSPM">;
297
298 def HasSPMX       :    Predicate<"Subtarget->hasSPMX()">,
299                          AssemblerPredicate<"FeatureSPMX">;
300
301 def HasDES        :    Predicate<"Subtarget->hasDES()">,
302                          AssemblerPredicate<"FeatureDES">;
303
304 def SupportsRMW   :    Predicate<"Subtarget->supportsRMW()">,
305                          AssemblerPredicate<"FeatureRMW">;
306
307 def SupportsMultiplication : Predicate<"Subtarget->supportsMultiplication()">,
308                                AssemblerPredicate<"FeatureMultiplication">;
309
310 def HasBREAK      :    Predicate<"Subtarget->hasBREAK()">,
311                          AssemblerPredicate<"FeatureBREAK">;
312
313 def HasTinyEncoding : Predicate<"Subtarget->hasTinyEncoding()">,
314                         AssemblerPredicate<"FeatureTinyEncoding">;
315
316
317 // AVR specific condition code. These correspond to AVR_*_COND in
318 // AVRInstrInfo.td. They must be kept in synch.
319 def AVR_COND_EQ : PatLeaf<(i8 0)>;
320 def AVR_COND_NE : PatLeaf<(i8 1)>;
321 def AVR_COND_GE : PatLeaf<(i8 2)>;
322 def AVR_COND_LT : PatLeaf<(i8 3)>;
323 def AVR_COND_SH : PatLeaf<(i8 4)>;
324 def AVR_COND_LO : PatLeaf<(i8 5)>;
325 def AVR_COND_MI : PatLeaf<(i8 6)>;
326 def AVR_COND_PL : PatLeaf<(i8 7)>;
327
328
329 //===----------------------------------------------------------------------===//
330 //===----------------------------------------------------------------------===//
331 // AVR Instruction list
332 //===----------------------------------------------------------------------===//
333 //===----------------------------------------------------------------------===//
334
335 // ADJCALLSTACKDOWN/UP implicitly use/def SP because they may be expanded into
336 // a stack adjustment and the codegen must know that they may modify the stack
337 // pointer before prolog-epilog rewriting occurs.
338 // Pessimistically assume ADJCALLSTACKDOWN / ADJCALLSTACKUP will become
339 // sub / add which can clobber SREG.
340 let Defs = [SP, SREG],
341 Uses = [SP] in
342 {
343   def ADJCALLSTACKDOWN : Pseudo<(outs),
344                                 (ins i16imm:$amt, i16imm:$amt2),
345                                 "#ADJCALLSTACKDOWN",
346                                 [(AVRcallseq_start timm:$amt, timm:$amt2)]>;
347
348   // R31R30 is used to update SP, since it is a scratch reg and this instruction
349   // is placed after the function call then R31R30 should be always free.
350   //let Defs = [R31R30],
351   //Uses = [R31R30] in
352   //:TODO: if we enable this, the pseudo is killed because it looks dead
353   def ADJCALLSTACKUP : Pseudo<(outs),
354                               (ins i16imm:$amt1, i16imm:$amt2),
355                               "#ADJCALLSTACKUP",
356                               [(AVRcallseq_end timm:$amt1, timm:$amt2)]>;
357 }
358
359 //===----------------------------------------------------------------------===//
360 // Addition
361 //===----------------------------------------------------------------------===//
362 let isCommutable = 1,
363 Constraints = "$src = $rd",
364 Defs = [SREG] in
365 {
366   // ADD Rd, Rr
367   // Adds two 8-bit registers.
368   def ADDRdRr : FRdRr<0b0000,
369                       0b11,
370                       (outs GPR8:$rd),
371                       (ins GPR8:$src, GPR8:$rr),
372                       "add\t$rd, $rr",
373                       [(set i8:$rd, (add i8:$src, i8:$rr)),
374                        (implicit SREG)]>;
375
376   // ADDW Rd+1:Rd, Rr+1:Rr
377   // Pseudo instruction to add four 8-bit registers as two 16-bit values.
378   //
379   // Expands to:
380   // add Rd,    Rr
381   // adc Rd+1, Rr+1
382   def ADDWRdRr : Pseudo<(outs DREGS:$rd),
383                         (ins DREGS:$src, DREGS:$rr),
384                         "addw\t$rd, $rr",
385                         [(set i16:$rd, (add i16:$src, i16:$rr)),
386                          (implicit SREG)]>;
387
388   // ADC Rd, Rr
389   // Adds two 8-bit registers with carry.
390   let Uses = [SREG] in
391   def ADCRdRr : FRdRr<0b0001,
392                       0b11,
393                       (outs GPR8:$rd),
394                       (ins GPR8:$src, GPR8:$rr),
395                       "adc\t$rd, $rr",
396                       [(set i8:$rd, (adde i8:$src, i8:$rr)),
397                        (implicit SREG)]>;
398
399   // ADCW Rd+1:Rd, Rr+1:Rr
400   // Pseudo instruction to add four 8-bit registers as two 16-bit values with
401   // carry.
402   //
403   // Expands to:
404   // adc Rd,   Rr
405   // adc Rd+1, Rr+1
406   let Uses = [SREG] in
407   def ADCWRdRr : Pseudo<(outs DREGS:$rd),
408                         (ins DREGS:$src, DREGS:$rr),
409                         "adcw\t$rd, $rr",
410                         [(set i16:$rd, (adde i16:$src, i16:$rr)),
411                          (implicit SREG)]>;
412
413   // AIDW Rd, k
414   // Adds an immediate 6-bit value K to Rd, placing the result in Rd.
415   def ADIWRdK : FWRdK<0b0,
416                       (outs IWREGS:$rd),
417                       (ins IWREGS:$src, imm_arith6:$k),
418                       "adiw\t$rd, $k",
419                       [(set i16:$rd, (add i16:$src, uimm6:$k)),
420                        (implicit SREG)]>,
421                 Requires<[HasADDSUBIW]>;
422 }
423
424 //===----------------------------------------------------------------------===//
425 // Subtraction
426 //===----------------------------------------------------------------------===//
427 let Constraints = "$src = $rd",
428 Defs = [SREG] in
429 {
430   // SUB Rd, Rr
431   // Subtracts the 8-bit value of Rr from Rd and places the value in Rd.
432   def SUBRdRr : FRdRr<0b0001,
433                       0b10,
434                       (outs GPR8:$rd),
435                       (ins GPR8:$src, GPR8:$rr),
436                       "sub\t$rd, $rr",
437                       [(set i8:$rd, (sub i8:$src, i8:$rr)),
438                        (implicit SREG)]>;
439
440   // SUBW Rd+1:Rd, Rr+1:Rr
441   // Subtracts two 16-bit values and places the result into Rd.
442   //
443   // Expands to:
444   // sub Rd,   Rr
445   // sbc Rd+1, Rr+1
446   def SUBWRdRr : Pseudo<(outs DREGS:$rd),
447                         (ins DREGS:$src, DREGS:$rr),
448                         "subw\t$rd, $rr",
449                         [(set i16:$rd, (sub i16:$src, i16:$rr)),
450                          (implicit SREG)]>;
451
452   def SUBIRdK : FRdK<0b0101,
453                      (outs LD8:$rd),
454                      (ins LD8:$src, imm_ldi8:$k),
455                      "subi\t$rd, $k",
456                      [(set i8:$rd, (sub i8:$src, imm:$k)),
457                       (implicit SREG)]>;
458
459   // SUBIW Rd+1:Rd, K+1:K
460   //
461   // Expands to:
462   // subi Rd,   K
463   // sbci Rd+1, K+1
464   def SUBIWRdK : Pseudo<(outs DLDREGS:$rd),
465                         (ins DLDREGS:$src, i16imm:$rr),
466                         "subiw\t$rd, $rr",
467                         [(set i16:$rd, (sub i16:$src, imm:$rr)),
468                          (implicit SREG)]>;
469
470   def SBIWRdK : FWRdK<0b1,
471                       (outs IWREGS:$rd),
472                       (ins IWREGS:$src, imm_arith6:$k),
473                       "sbiw\t$rd, $k",
474                       [(set i16:$rd, (sub i16:$src, uimm6:$k)),
475                        (implicit SREG)]>,
476                 Requires<[HasADDSUBIW]>;
477
478   // Subtract with carry operations which must read the carry flag in SREG.
479   let Uses = [SREG] in
480   {
481     def SBCRdRr : FRdRr<0b0000,
482                         0b10,
483                         (outs GPR8:$rd),
484                         (ins GPR8:$src, GPR8:$rr),
485                         "sbc\t$rd, $rr",
486                         [(set i8:$rd, (sube i8:$src, i8:$rr)),
487                          (implicit SREG)]>;
488
489     // SBCW Rd+1:Rd, Rr+1:Rr
490     //
491     // Expands to:
492     // sbc Rd,   Rr
493     // sbc Rd+1, Rr+1
494     def SBCWRdRr : Pseudo<(outs DREGS:$rd),
495                           (ins DREGS:$src, DREGS:$rr),
496                           "sbcw\t$rd, $rr",
497                           [(set i16:$rd, (sube i16:$src, i16:$rr)),
498                            (implicit SREG)]>;
499
500     def SBCIRdK : FRdK<0b0100,
501                        (outs LD8:$rd),
502                        (ins LD8:$src, imm_ldi8:$k),
503                        "sbci\t$rd, $k",
504                        [(set i8:$rd, (sube i8:$src, imm:$k)),
505                         (implicit SREG)]>;
506
507     // SBCIW Rd+1:Rd, K+1:K
508     // sbci Rd,   K
509     // sbci Rd+1, K+1
510     def SBCIWRdK : Pseudo<(outs DLDREGS:$rd),
511                           (ins DLDREGS:$src, i16imm:$rr),
512                           "sbciw\t$rd, $rr",
513                           [(set i16:$rd, (sube i16:$src, imm:$rr)),
514                            (implicit SREG)]>;
515   }
516 }
517
518 //===----------------------------------------------------------------------===//
519 // Increment and Decrement
520 //===----------------------------------------------------------------------===//
521 let Constraints = "$src = $rd",
522 Defs = [SREG] in
523 {
524   def INCRd : FRd<0b1001,
525                   0b0100011,
526                   (outs GPR8:$rd),
527                   (ins GPR8:$src),
528                   "inc\t$rd",
529                   [(set i8:$rd, (add i8:$src, 1)), (implicit SREG)]>;
530
531   def DECRd : FRd<0b1001,
532                   0b0101010,
533                   (outs GPR8:$rd),
534                   (ins GPR8:$src),
535                   "dec\t$rd",
536                   [(set i8:$rd, (add i8:$src, -1)), (implicit SREG)]>;
537 }
538
539 //===----------------------------------------------------------------------===//
540 // Multiplication
541 //===----------------------------------------------------------------------===//
542
543 let isCommutable = 1,
544 Defs = [R1, R0, SREG] in
545 {
546   // MUL Rd, Rr
547   // Multiplies Rd by Rr and places the result into R1:R0.
548   let usesCustomInserter = 1 in {
549     def MULRdRr : FRdRr<0b1001, 0b11,
550                         (outs),
551                         (ins GPR8:$lhs, GPR8:$rhs),
552                         "mul\t$lhs, $rhs",
553                         [/*(set R1, R0, (smullohi i8:$lhs, i8:$rhs))*/]>,
554                     Requires<[SupportsMultiplication]>;
555
556     def MULSRdRr : FMUL2RdRr<0,
557                              (outs),
558                              (ins GPR8:$lhs, GPR8:$rhs),
559                              "muls\t$lhs, $rhs",
560                              []>,
561                    Requires<[SupportsMultiplication]>;
562   }
563
564   def MULSURdRr : FMUL2RdRr<1,
565                             (outs),
566                             (ins GPR8:$lhs, GPR8:$rhs),
567                             "mulsu\t$lhs, $rhs",
568                             []>,
569                   Requires<[SupportsMultiplication]>;
570
571   def FMUL : FFMULRdRr<0b01,
572                        (outs),
573                        (ins GPR8:$lhs, GPR8:$rhs),
574                        "fmul\t$lhs, $rhs",
575                        []>,
576              Requires<[SupportsMultiplication]>;
577
578   def FMULS : FFMULRdRr<0b10,
579                         (outs),
580                         (ins GPR8:$lhs, GPR8:$rhs),
581                         "fmuls\t$lhs, $rhs",
582                         []>,
583               Requires<[SupportsMultiplication]>;
584
585   def FMULSU : FFMULRdRr<0b11,
586                          (outs),
587                          (ins GPR8:$lhs, GPR8:$rhs),
588                          "fmulsu\t$lhs, $rhs",
589                          []>,
590                Requires<[SupportsMultiplication]>;
591 }
592
593 let Defs = [R15, R14, R13, R12, R11, R10, R9,
594             R8, R7, R6, R5, R4, R3, R2, R1, R0] in
595 def DESK : FDES<(outs),
596                 (ins i8imm:$k),
597                 "des\t$k",
598                 []>,
599            Requires<[HasDES]>;
600
601 //===----------------------------------------------------------------------===//
602 // Logic
603 //===----------------------------------------------------------------------===//
604 let Constraints = "$src = $rd",
605 Defs = [SREG] in
606 {
607   // Register-Register logic instructions (which have the
608   // property of commutativity).
609   let isCommutable = 1 in
610   {
611     def ANDRdRr : FRdRr<0b0010,
612                         0b00,
613                         (outs GPR8:$rd),
614                         (ins GPR8:$src, GPR8:$rr),
615                         "and\t$rd, $rr",
616                         [(set i8:$rd, (and i8:$src, i8:$rr)),
617                          (implicit SREG)]>;
618
619     // ANDW Rd+1:Rd, Rr+1:Rr
620     //
621     // Expands to:
622     // and Rd,   Rr
623     // and Rd+1, Rr+1
624     def ANDWRdRr : Pseudo<(outs DREGS:$rd),
625                           (ins DREGS:$src, DREGS:$rr),
626                           "andw\t$rd, $rr",
627                           [(set i16:$rd, (and i16:$src, i16:$rr)),
628                            (implicit SREG)]>;
629
630     def ORRdRr : FRdRr<0b0010,
631                        0b10,
632                        (outs GPR8:$rd),
633                        (ins GPR8:$src, GPR8:$rr),
634                        "or\t$rd, $rr",
635                        [(set i8:$rd, (or i8:$src, i8:$rr)),
636                         (implicit SREG)]>;
637
638     // ORW Rd+1:Rd, Rr+1:Rr
639     //
640     // Expands to:
641     // or Rd,   Rr
642     // or Rd+1, Rr+1
643     def ORWRdRr : Pseudo<(outs DREGS:$rd),
644                          (ins DREGS:$src, DREGS:$rr),
645                          "orw\t$rd, $rr",
646                          [(set i16:$rd, (or i16:$src, i16:$rr)),
647                           (implicit SREG)]>;
648
649     def EORRdRr : FRdRr<0b0010,
650                         0b01,
651                         (outs GPR8:$rd),
652                         (ins GPR8:$src, GPR8:$rr),
653                         "eor\t$rd, $rr",
654                         [(set i8:$rd, (xor i8:$src, i8:$rr)),
655                          (implicit SREG)]>;
656
657     // EORW Rd+1:Rd, Rr+1:Rr
658     //
659     // Expands to:
660     // eor Rd,   Rr
661     // eor Rd+1, Rr+1
662     def EORWRdRr : Pseudo<(outs DREGS:$rd),
663                           (ins DREGS:$src, DREGS:$rr),
664                           "eorw\t$rd, $rr",
665                           [(set i16:$rd, (xor i16:$src, i16:$rr)),
666                            (implicit SREG)]>;
667   }
668
669   def ANDIRdK : FRdK<0b0111,
670                      (outs LD8:$rd),
671                      (ins LD8:$src, imm_ldi8:$k),
672                      "andi\t$rd, $k",
673                      [(set i8:$rd, (and i8:$src, imm:$k)),
674                       (implicit SREG)]>;
675
676   // ANDI Rd+1:Rd, K+1:K
677   //
678   // Expands to:
679   // andi Rd,   K
680   // andi Rd+1, K+1
681   def ANDIWRdK : Pseudo<(outs DLDREGS:$rd),
682                         (ins DLDREGS:$src, i16imm:$k),
683                         "andiw\t$rd, $k",
684                         [(set i16:$rd, (and i16:$src, imm:$k)),
685                          (implicit SREG)]>;
686
687   def ORIRdK : FRdK<0b0110,
688                     (outs LD8:$rd),
689                     (ins LD8:$src, imm_ldi8:$k),
690                     "ori\t$rd, $k",
691                     [(set i8:$rd, (or i8:$src, imm:$k)),
692                      (implicit SREG)]>;
693
694   // ORIW Rd+1:Rd, K+1,K
695   //
696   // Expands to:
697   // ori Rd,   K
698   // ori Rd+1, K+1
699   def ORIWRdK : Pseudo<(outs DLDREGS:$rd),
700                        (ins DLDREGS:$src, i16imm:$rr),
701                        "oriw\t$rd, $rr",
702                        [(set i16:$rd, (or i16:$src, imm:$rr)),
703                         (implicit SREG)]>;
704 }
705
706 //===----------------------------------------------------------------------===//
707 // One's/Two's Complement
708 //===----------------------------------------------------------------------===//
709 let Constraints = "$src = $rd",
710 Defs = [SREG] in
711 {
712   def COMRd : FRd<0b1001,
713                   0b0100000,
714                   (outs GPR8:$rd),
715                   (ins GPR8:$src),
716                   "com\t$rd",
717                   [(set i8:$rd, (not i8:$src)), (implicit SREG)]>;
718
719   // COMW Rd+1:Rd
720   //
721   // Expands to:
722   // com Rd
723   // com Rd+1
724   def COMWRd : Pseudo<(outs DREGS:$rd),
725                       (ins DREGS:$src),
726                       "comw\t$rd",
727                       [(set i16:$rd, (not i16:$src)), (implicit SREG)]>;
728
729   //:TODO: optimize NEG for wider types
730   def NEGRd : FRd<0b1001,
731                   0b0100001,
732                   (outs GPR8:$rd),
733                   (ins GPR8:$src),
734                   "neg\t$rd",
735                   [(set i8:$rd, (ineg i8:$src)), (implicit SREG)]>;
736 }
737
738 // TST Rd
739 // Test for zero of minus.
740 // This operation is identical to a `Rd AND Rd`.
741 def : InstAlias<"tst\t$rd", (ANDRdRr GPR8:$rd, GPR8:$rd)>;
742
743 // SBR Rd, K
744 //
745 // Mnemonic alias to 'ORI Rd, K'. Same bit pattern, same operands,
746 // same everything.
747 def : InstAlias<"sbr\t$rd, $k",
748                 (ORIRdK LD8:$rd, imm_ldi8:$k),
749                 /* Disable display, so we don't override ORI */ 0>;
750
751 //===----------------------------------------------------------------------===//
752 // Jump instructions
753 //===----------------------------------------------------------------------===//
754 let isBarrier = 1,
755 isBranch = 1,
756 isTerminator = 1 in
757 {
758   def RJMPk : FBRk<0,
759                    (outs),
760                    (ins brtarget_13:$target),
761                    "rjmp\t$target",
762                    [(br bb:$target)]>;
763
764   let isIndirectBranch = 1,
765   Uses = [R31R30] in
766   def IJMP : F16<0b1001010000001001,
767                  (outs),
768                  (ins),
769                  "ijmp",
770                  []>,
771              Requires<[HasIJMPCALL]>;
772
773   let isIndirectBranch = 1,
774   Uses = [R31R30] in
775   def EIJMP : F16<0b1001010000011001,
776                   (outs),
777                   (ins),
778                   "eijmp",
779                   []>,
780               Requires<[HasEIJMPCALL]>;
781
782   def JMPk : F32BRk<0b110,
783                     (outs),
784                     (ins call_target:$k),
785                     "jmp\t$k",
786                     []>,
787              Requires<[HasJMPCALL]>;
788 }
789
790 //===----------------------------------------------------------------------===//
791 // Call instructions
792 //===----------------------------------------------------------------------===//
793 let isCall = 1 in
794 {
795   // SP is marked as a use to prevent stack-pointer assignments that appear
796   // immediately before calls from potentially appearing dead.
797   let Uses = [SP] in
798   def RCALLk : FBRk<1,
799                     (outs),
800                     (ins brtarget_13:$target),
801                     "rcall\t$target",
802                     []>;
803
804   // SP is marked as a use to prevent stack-pointer assignments that appear
805   // immediately before calls from potentially appearing dead.
806   let Uses = [SP, R31R30] in
807   def ICALL : F16<0b1001010100001001,
808                   (outs),
809                   (ins variable_ops),
810                   "icall",
811                   []>,
812               Requires<[HasIJMPCALL]>;
813
814   // SP is marked as a use to prevent stack-pointer assignments that appear
815   // immediately before calls from potentially appearing dead.
816   let Uses = [SP, R31R30] in
817   def EICALL : F16<0b1001010100011001,
818                    (outs),
819                    (ins variable_ops),
820                    "eicall",
821                    []>,
822                Requires<[HasEIJMPCALL]>;
823
824   // SP is marked as a use to prevent stack-pointer assignments that appear
825   // immediately before calls from potentially appearing dead.
826   //
827   //:TODO: the imm field can be either 16 or 22 bits in devices with more
828   // than 64k of ROM, fix it once we support the largest devices.
829   let Uses = [SP] in
830   def CALLk : F32BRk<0b111,
831                      (outs),
832                      (ins call_target:$k),
833                      "call\t$k",
834                      [(AVRcall imm:$k)]>,
835               Requires<[HasJMPCALL]>;
836 }
837
838 //===----------------------------------------------------------------------===//
839 // Return instructions.
840 //===----------------------------------------------------------------------===//
841 let isTerminator = 1,
842 isReturn = 1,
843 isBarrier = 1 in 
844 {
845   def RET : F16<0b1001010100001000,
846                 (outs),
847                 (ins),
848                 "ret",
849                 [(AVRretflag)]>;
850
851   def RETI : F16<0b1001010100011000,
852                  (outs),
853                  (ins),
854                  "reti",
855                  [(AVRretiflag)]>;
856 }
857
858 //===----------------------------------------------------------------------===//
859 // Compare operations.
860 //===----------------------------------------------------------------------===//
861 let Defs = [SREG] in
862 {
863   // CPSE Rd, Rr
864   // Compare Rd and Rr, skipping the next instruction if they are equal.
865   let isBarrier = 1,
866   isBranch = 1,
867   isTerminator = 1 in
868   def CPSE : FRdRr<0b0001,
869                    0b00,
870                    (outs),
871                    (ins GPR8:$rd, GPR8:$rr),
872                    "cpse\t$rd, $rr",
873                    []>;
874
875   def CPRdRr : FRdRr<0b0001,
876                      0b01,
877                      (outs),
878                      (ins GPR8:$rd, GPR8:$rr),
879                      "cp\t$rd, $rr",
880                      [(AVRcmp i8:$rd, i8:$rr), (implicit SREG)]>;
881
882   // CPW Rd+1:Rd, Rr+1:Rr
883   //
884   // Expands to:
885   // cp  Rd,   Rr
886   // cpc Rd+1, Rr+1
887   def CPWRdRr : Pseudo<(outs),
888                        (ins DREGS:$src, DREGS:$src2),
889                        "cpw\t$src, $src2",
890                        [(AVRcmp i16:$src, i16:$src2), (implicit SREG)]>;
891
892   let Uses = [SREG] in
893   def CPCRdRr : FRdRr<0b0000,
894                       0b01,
895                       (outs),
896                       (ins GPR8:$rd, GPR8:$rr),
897                       "cpc\t$rd, $rr",
898                       [(AVRcmpc i8:$rd, i8:$rr), (implicit SREG)]>;
899
900   // CPCW Rd+1:Rd. Rr+1:Rr
901   //
902   // Expands to:
903   // cpc Rd,   Rr
904   // cpc Rd+1, Rr+1
905   let Uses = [SREG] in
906   def CPCWRdRr : Pseudo<(outs),
907                         (ins DREGS:$src, DREGS:$src2),
908                         "cpcw\t$src, $src2",
909                         [(AVRcmpc i16:$src, i16:$src2), (implicit SREG)]>;
910
911   // CPI Rd, K
912   // Compares a register with an 8 bit immediate.
913   def CPIRdK : FRdK<0b0011,
914                     (outs),
915                     (ins LD8:$rd, imm_ldi8:$k),
916                     "cpi\t$rd, $k",
917                     [(AVRcmp i8:$rd, imm:$k), (implicit SREG)]>;
918 }
919
920 //===----------------------------------------------------------------------===//
921 // Register conditional skipping/branching operations.
922 //===----------------------------------------------------------------------===//
923 let isBranch = 1,
924 isTerminator = 1 in
925 {
926   // Conditional skipping on GPR register bits, and
927   // conditional skipping on IO register bits.
928   let isBarrier = 1 in
929   {
930     def SBRCRrB : FRdB<0b10,
931                        (outs),
932                        (ins GPR8:$rr, i8imm:$b),
933                        "sbrc\t$rr, $b",
934                        []>;
935
936     def SBRSRrB : FRdB<0b11,
937                        (outs),
938                        (ins GPR8:$rr, i8imm:$b),
939                        "sbrs\t$rr, $b",
940                        []>;
941
942     def SBICAb : FIOBIT<0b01,
943                         (outs),
944                         (ins imm_port5:$a, i8imm:$b),
945                         "sbic\t$a, $b",
946                         []>;
947
948     def SBISAb : FIOBIT<0b11,
949                         (outs),
950                         (ins imm_port5:$a, i8imm:$b),
951                         "sbis\t$a, $b",
952                         []>;
953   }
954
955   // Relative branches on status flag bits.
956   let Uses = [SREG] in
957   {
958     // BRBS s, k
959     // Branch if `s` flag in status register is set.
960     def BRBSsk : FSK<0,
961                      (outs),
962                      (ins i8imm:$s, relbrtarget_7:$k),
963                      "brbs\t$s, $k",
964                      []>;
965
966     // BRBC s, k
967     // Branch if `s` flag in status register is clear.
968     def BRBCsk : FSK<1,
969                      (outs),
970                      (ins i8imm:$s, relbrtarget_7:$k),
971                      "brbc\t$s, $k",
972                      []>;
973   }
974 }
975
976
977 // BRCS k
978 // Branch if carry flag is set
979 def : InstAlias<"brcs\t$k", (BRBSsk 0, relbrtarget_7:$k)>;
980
981 // BRCC k
982 // Branch if carry flag is clear
983 def : InstAlias<"brcc\t$k", (BRBCsk 0, relbrtarget_7:$k)>;
984
985 // BRHS k
986 // Branch if half carry flag is set
987 def : InstAlias<"brhs\t$k", (BRBSsk 5, relbrtarget_7:$k)>;
988
989 // BRHC k
990 // Branch if half carry flag is clear
991 def : InstAlias<"brhc\t$k", (BRBCsk 5, relbrtarget_7:$k)>;
992
993 // BRTS k
994 // Branch if the T flag is set
995 def : InstAlias<"brts\t$k", (BRBSsk 6, relbrtarget_7:$k)>;
996
997 // BRTC k
998 // Branch if the T flag is clear
999 def : InstAlias<"brtc\t$k", (BRBCsk 6, relbrtarget_7:$k)>;
1000
1001 // BRVS k
1002 // Branch if the overflow flag is set
1003 def : InstAlias<"brvs\t$k", (BRBSsk 3, relbrtarget_7:$k)>;
1004
1005 // BRVC k
1006 // Branch if the overflow flag is clear
1007 def : InstAlias<"brvc\t$k", (BRBCsk 3, relbrtarget_7:$k)>;
1008
1009 // BRIE k
1010 // Branch if the global interrupt flag is enabled
1011 def : InstAlias<"brie\t$k", (BRBSsk 7, relbrtarget_7:$k)>;
1012
1013 // BRID k
1014 // Branch if the global interrupt flag is disabled
1015 def : InstAlias<"brid\t$k", (BRBCsk 7, relbrtarget_7:$k)>;
1016
1017 //===----------------------------------------------------------------------===//
1018 // PC-relative conditional branches
1019 //===----------------------------------------------------------------------===//
1020 // Based on status register. We cannot simplify these into instruction aliases
1021 // because we also need to be able to specify a pattern to match for ISel.
1022 let isBranch = 1,
1023 isTerminator = 1,
1024 Uses = [SREG] in
1025 {
1026   def BREQk : FBRsk<0,
1027                     0b001,
1028                     (outs),
1029                     (ins relbrtarget_7:$target),
1030                     "breq\t$target",
1031                     [(AVRbrcond bb:$target, AVR_COND_EQ)]>;
1032
1033   def BRNEk : FBRsk<1,
1034                     0b001,
1035                     (outs),
1036                     (ins relbrtarget_7:$target),
1037                     "brne\t$target",
1038                     [(AVRbrcond bb:$target, AVR_COND_NE)]>;
1039
1040
1041   def BRSHk : FBRsk<1,
1042                     0b000,
1043                     (outs),
1044                     (ins relbrtarget_7:$target),
1045                     "brsh\t$target",
1046                     [(AVRbrcond bb:$target, AVR_COND_SH)]>;
1047
1048   def BRLOk : FBRsk<0,
1049                     0b000,
1050                     (outs),
1051                     (ins relbrtarget_7:$target),
1052                     "brlo\t$target",
1053                     [(AVRbrcond bb:$target, AVR_COND_LO)]>;
1054
1055   def BRMIk : FBRsk<0,
1056                     0b010,
1057                     (outs),
1058                     (ins relbrtarget_7:$target),
1059                     "brmi\t$target",
1060                     [(AVRbrcond bb:$target, AVR_COND_MI)]>;
1061
1062   def BRPLk : FBRsk<1,
1063                     0b010,
1064                     (outs),
1065                     (ins relbrtarget_7:$target),
1066                     "brpl\t$target",
1067                     [(AVRbrcond bb:$target, AVR_COND_PL)]>;
1068
1069   def BRGEk : FBRsk<1,
1070                     0b100,
1071                     (outs),
1072                     (ins relbrtarget_7:$target),
1073                     "brge\t$target",
1074                     [(AVRbrcond bb:$target, AVR_COND_GE)]>;
1075
1076   def BRLTk : FBRsk<0,
1077                     0b100,
1078                     (outs),
1079                     (ins relbrtarget_7:$target),
1080                     "brlt\t$target",
1081                     [(AVRbrcond bb:$target, AVR_COND_LT)]>;
1082 }
1083
1084 //===----------------------------------------------------------------------===//
1085 // Data transfer instructions
1086 //===----------------------------------------------------------------------===//
1087 // 8 and 16-bit register move instructions.
1088 let hasSideEffects = 0 in
1089 {
1090   def MOVRdRr : FRdRr<0b0010,
1091                       0b11,
1092                       (outs GPR8:$rd),
1093                       (ins GPR8:$rr),
1094                       "mov\t$rd, $rr",
1095                       []>;
1096
1097   def MOVWRdRr : FMOVWRdRr<(outs DREGS:$dst),
1098                            (ins DREGS:$src),
1099                            "movw\t$dst, $src",
1100                            []>,
1101                  Requires<[HasMOVW]>;
1102 }
1103
1104 // Load immediate values into registers.
1105 let isReMaterializable = 1 in
1106 {
1107   def LDIRdK : FRdK<0b1110,
1108                     (outs LD8:$rd),
1109                     (ins imm_ldi8:$k),
1110                     "ldi\t$rd, $k",
1111                     [(set i8:$rd, imm:$k)]>;
1112
1113   // LDIW Rd+1:Rd, K+1:K
1114   //
1115   // Expands to:
1116   // ldi Rd,   K
1117   // ldi Rd+1, K+1
1118   def LDIWRdK : Pseudo<(outs DLDREGS:$dst),
1119                        (ins i16imm:$src),
1120                        "ldiw\t$dst, $src",
1121                        [(set i16:$dst, imm:$src)]>;
1122 }
1123
1124 // Load from data space into register.
1125 let canFoldAsLoad = 1,
1126 isReMaterializable = 1 in
1127 {
1128   def LDSRdK : F32DM<0b0,
1129                      (outs GPR8:$rd),
1130                      (ins imm16:$k),
1131                      "lds\t$rd, $k",
1132                      [(set i8:$rd, (load imm:$k))]>,
1133                Requires<[HasSRAM]>;
1134
1135   // LDSW Rd+1:Rd, K+1:K
1136   //
1137   // Expands to:
1138   // lds Rd,  (K+1:K)
1139   // lds Rd+1 (K+1:K) + 1
1140   def LDSWRdK : Pseudo<(outs DREGS:$dst),
1141                        (ins i16imm:$src),
1142                        "ldsw\t$dst, $src",
1143                        [(set i16:$dst, (load imm:$src))]>,
1144                 Requires<[HasSRAM]>;
1145 }
1146
1147 // Indirect loads.
1148 let canFoldAsLoad = 1,
1149 isReMaterializable = 1 in
1150 {
1151   def LDRdPtr : FSTLD<0,
1152                       0b00,
1153                       (outs GPR8:$reg),
1154                       (ins LDSTPtrReg:$ptrreg),
1155                       "ld\t$reg, $ptrreg",
1156                       [(set GPR8:$reg, (load i16:$ptrreg))]>,
1157                 Requires<[HasSRAM]>;
1158
1159   // LDW Rd+1:Rd, P
1160   //
1161   // Expands to:
1162   // ld  Rd,   P
1163   // ldd Rd+1, P+1
1164   let Constraints = "@earlyclobber $reg" in
1165   def LDWRdPtr : Pseudo<(outs DREGS:$reg),
1166                         (ins PTRDISPREGS:$ptrreg),
1167                         "ldw\t$reg, $ptrreg",
1168                         [(set i16:$reg, (load i16:$ptrreg))]>,
1169                  Requires<[HasSRAM]>;
1170 }
1171
1172 // Indirect loads (with postincrement or predecrement).
1173 let mayLoad = 1,
1174 hasSideEffects = 0,
1175 Constraints = "$ptrreg = $base_wb,@earlyclobber $reg" in
1176 {
1177   def LDRdPtrPi : FSTLD<0,
1178                         0b01,
1179                         (outs GPR8:$reg, PTRREGS:$base_wb),
1180                         (ins LDSTPtrReg:$ptrreg),
1181                         "ld\t$reg, $ptrreg+",
1182                         []>,
1183                   Requires<[HasSRAM]>;
1184
1185   // LDW Rd+1:Rd, P+
1186   // Expands to:
1187   // ld Rd,   P+
1188   // ld Rd+1, P+
1189   def LDWRdPtrPi : Pseudo<(outs DREGS:$reg, PTRREGS:$base_wb),
1190                           (ins PTRREGS:$ptrreg),
1191                           "ldw\t$reg, $ptrreg+",
1192                           []>,
1193                    Requires<[HasSRAM]>;
1194
1195   def LDRdPtrPd : FSTLD<0,
1196                         0b10,
1197                         (outs GPR8:$reg, PTRREGS:$base_wb),
1198                         (ins LDSTPtrReg:$ptrreg),
1199                         "ld\t$reg, -$ptrreg",
1200                         []>,
1201                   Requires<[HasSRAM]>;
1202
1203   // LDW Rd+1:Rd, -P
1204   //
1205   // Expands to:
1206   // ld Rd+1, -P
1207   // ld Rd,   -P
1208   def LDWRdPtrPd : Pseudo<(outs DREGS:$reg, PTRREGS:$base_wb),
1209                           (ins PTRREGS:$ptrreg),
1210                           "ldw\t$reg, -$ptrreg",
1211                           []>,
1212                    Requires<[HasSRAM]>;
1213 }
1214
1215 // Load indirect with displacement operations.
1216 let canFoldAsLoad = 1,
1217 isReMaterializable = 1 in
1218 {
1219   let Constraints = "@earlyclobber $reg" in
1220   def LDDRdPtrQ : FSTDLDD<0,
1221                           (outs GPR8:$reg),
1222                           (ins memri:$memri),
1223                           "ldd\t$reg, $memri",
1224                           [(set i8:$reg, (load addr:$memri))]>,
1225                   Requires<[HasSRAM]>;
1226
1227   // LDDW Rd+1:Rd, P+q
1228   //
1229   // Expands to:
1230   // ldd Rd,   P+q
1231   // ldd Rd+1, P+q+1
1232   let Constraints = "@earlyclobber $dst" in
1233   def LDDWRdPtrQ : Pseudo<(outs DREGS_WITHOUT_YZ_WORKAROUND:$dst),
1234                           (ins memri:$memri),
1235                           "lddw\t$dst, $memri",
1236                           [(set i16:$dst, (load addr:$memri))]>,
1237                    Requires<[HasSRAM]>;
1238
1239   // An identical pseudo instruction to LDDWRdPtrQ, expect restricted to the Y
1240   // register and without the @earlyclobber flag.
1241   //
1242   // Used to work around a bug caused by the register allocator not
1243   // being able to handle the expansion of a COPY into an machine instruction
1244   // that has an earlyclobber flag. This is because the register allocator will
1245   // try expand a copy from a register slot into an earlyclobber instruction.
1246   // Instructions that are earlyclobber need to be in a dedicated earlyclobber slot.
1247   //
1248   // This pseudo instruction can be used pre-AVR pseudo expansion in order to
1249   // get a frame index load without directly using earlyclobber instructions.
1250   //
1251   // The pseudo expansion pass trivially expands this into LDDWRdPtrQ.
1252   //
1253   // This instruction may be removed once PR13375 is fixed.
1254   let mayLoad = 1,
1255   hasSideEffects = 0 in
1256   def LDDWRdYQ : Pseudo<(outs DREGS:$dst),
1257                         (ins memri:$memri),
1258                         "lddw\t$dst, $memri",
1259                         []>,
1260                  Requires<[HasSRAM]>;
1261 }
1262
1263 class AtomicLoad<PatFrag Op, RegisterClass DRC,
1264                  RegisterClass PTRRC> :
1265   Pseudo<(outs DRC:$rd), (ins PTRRC:$rr), "atomic_op",
1266          [(set DRC:$rd, (Op i16:$rr))]>;
1267
1268 class AtomicStore<PatFrag Op, RegisterClass DRC,
1269                   RegisterClass PTRRC> :
1270   Pseudo<(outs), (ins PTRRC:$rd, DRC:$rr), "atomic_op",
1271          [(Op i16:$rd, DRC:$rr)]>;
1272
1273 class AtomicLoadOp<PatFrag Op, RegisterClass DRC,
1274                    RegisterClass PTRRC> :
1275   Pseudo<(outs DRC:$rd), (ins PTRRC:$rr, DRC:$operand),
1276          "atomic_op",
1277          [(set DRC:$rd, (Op i16:$rr, DRC:$operand))]>;
1278
1279 // FIXME: I think 16-bit atomic binary ops need to mark
1280 // r0 as clobbered.
1281
1282 // Atomic instructions
1283 // ===================
1284 //
1285 // These are all expanded by AVRExpandPseudoInsts
1286 //
1287 // 8-bit operations can use any pointer register because
1288 // they are expanded directly into an LD/ST instruction.
1289 //
1290 // 16-bit operations use 16-bit load/store postincrement instructions,
1291 // which require PTRDISPREGS.
1292
1293 def AtomicLoad8   : AtomicLoad<atomic_load_8, GPR8, PTRREGS>;
1294 def AtomicLoad16  : AtomicLoad<atomic_load_16, DREGS, PTRDISPREGS>;
1295
1296 def AtomicStore8  : AtomicStore<atomic_store_8, GPR8, PTRREGS>;
1297 def AtomicStore16 : AtomicStore<atomic_store_16, DREGS, PTRDISPREGS>;
1298
1299 class AtomicLoadOp8<PatFrag Op> : AtomicLoadOp<Op, GPR8, PTRREGS>;
1300 class AtomicLoadOp16<PatFrag Op> : AtomicLoadOp<Op, DREGS, PTRDISPREGS>;
1301
1302 def AtomicLoadAdd8  : AtomicLoadOp8<atomic_load_add_8>;
1303 def AtomicLoadAdd16 : AtomicLoadOp16<atomic_load_add_16>;
1304 def AtomicLoadSub8  : AtomicLoadOp8<atomic_load_sub_8>;
1305 def AtomicLoadSub16 : AtomicLoadOp16<atomic_load_sub_16>;
1306 def AtomicLoadAnd8  : AtomicLoadOp8<atomic_load_and_8>;
1307 def AtomicLoadAnd16 : AtomicLoadOp16<atomic_load_and_16>;
1308 def AtomicLoadOr8   : AtomicLoadOp8<atomic_load_or_8>;
1309 def AtomicLoadOr16  : AtomicLoadOp16<atomic_load_or_16>;
1310 def AtomicLoadXor8  : AtomicLoadOp8<atomic_load_xor_8>;
1311 def AtomicLoadXor16 : AtomicLoadOp16<atomic_load_xor_16>;
1312 def AtomicFence     : Pseudo<(outs), (ins), "atomic_fence",
1313                              [(atomic_fence timm, timm)]>;
1314
1315 // Indirect store from register to data space.
1316 def STSKRr : F32DM<0b1,
1317                    (outs),
1318                    (ins imm16:$k, GPR8:$rd),
1319                    "sts\t$k, $rd",
1320                    [(store i8:$rd, imm:$k)]>,
1321              Requires<[HasSRAM]>;
1322
1323 // STSW K+1:K, Rr+1:Rr
1324 //
1325 // Expands to:
1326 // sts Rr+1, (K+1:K) + 1
1327 // sts Rr,   (K+1:K)
1328 def STSWKRr : Pseudo<(outs),
1329                      (ins i16imm:$dst, DREGS:$src),
1330                      "stsw\t$dst, $src",
1331                      [(store i16:$src, imm:$dst)]>,
1332               Requires<[HasSRAM]>;
1333
1334 // Indirect stores.
1335 // ST P, Rr
1336 // Stores the value of Rr into the location addressed by pointer P.
1337 def STPtrRr : FSTLD<1,
1338                     0b00,
1339                     (outs),
1340                     (ins LDSTPtrReg:$ptrreg, GPR8:$reg),
1341                     "st\t$ptrreg, $reg",
1342                     [(store GPR8:$reg, i16:$ptrreg)]>,
1343               Requires<[HasSRAM]>;
1344
1345 // STW P, Rr+1:Rr
1346 // Stores the value of Rr into the location addressed by pointer P.
1347 //
1348 // Expands to:
1349 // st P, Rr
1350 // std P+1, Rr+1
1351 def STWPtrRr : Pseudo<(outs),
1352                       (ins PTRDISPREGS:$ptrreg, DREGS:$reg),
1353                       "stw\t$ptrreg, $reg",
1354                       [(store i16:$reg, i16:$ptrreg)]>,
1355                Requires<[HasSRAM]>;
1356
1357 // Indirect stores (with postincrement or predecrement).
1358 let Constraints = "$ptrreg = $base_wb,@earlyclobber $base_wb" in
1359 {
1360
1361   // ST P+, Rr
1362   // Stores the value of Rr into the location addressed by pointer P.
1363   // Post increments P.
1364   def STPtrPiRr : FSTLD<1,
1365                         0b01,
1366                         (outs LDSTPtrReg:$base_wb),
1367                         (ins LDSTPtrReg:$ptrreg, GPR8:$reg, i8imm:$offs),
1368                         "st\t$ptrreg+, $reg",
1369                         [(set i16:$base_wb,
1370                          (post_store GPR8:$reg, i16:$ptrreg, imm:$offs))]>,
1371                   Requires<[HasSRAM]>;
1372
1373   // STW P+, Rr+1:Rr
1374   // Stores the value of Rr into the location addressed by pointer P.
1375   // Post increments P.
1376   //
1377   // Expands to:
1378   // st P+, Rr
1379   // st P+, Rr+1
1380   def STWPtrPiRr : Pseudo<(outs PTRREGS:$base_wb),
1381                           (ins PTRREGS:$ptrreg, DREGS:$trh, i8imm:$offs),
1382                           "stw\t$ptrreg+, $trh",
1383                           [(set PTRREGS:$base_wb,
1384                            (post_store DREGS:$trh, PTRREGS:$ptrreg, imm:$offs))]>,
1385                    Requires<[HasSRAM]>;
1386
1387   // ST -P, Rr
1388   // Stores the value of Rr into the location addressed by pointer P.
1389   // Pre decrements P.
1390   def STPtrPdRr : FSTLD<1,
1391                         0b10,
1392                         (outs LDSTPtrReg:$base_wb),
1393                         (ins LDSTPtrReg:$ptrreg, GPR8:$reg, i8imm:$offs),
1394                         "st\t-$ptrreg, $reg",
1395                         [(set i16:$base_wb,
1396                          (pre_store GPR8:$reg, i16:$ptrreg, imm:$offs))]>,
1397                   Requires<[HasSRAM]>;
1398
1399   // STW -P, Rr+1:Rr
1400   // Stores the value of Rr into the location addressed by pointer P.
1401   // Pre decrements P.
1402   //
1403   // Expands to:
1404   // st -P, Rr+1
1405   // st -P, Rr
1406   def STWPtrPdRr : Pseudo<(outs PTRREGS:$base_wb),
1407                           (ins PTRREGS:$ptrreg, DREGS:$reg, i8imm:$offs),
1408                           "stw\t-$ptrreg, $reg",
1409                           [(set PTRREGS:$base_wb,
1410                            (pre_store i16:$reg, i16:$ptrreg, imm:$offs))]>,
1411                    Requires<[HasSRAM]>;
1412 }
1413
1414 // Store indirect with displacement operations.
1415 // STD P+q, Rr
1416 // Stores the value of Rr into the location addressed by pointer P with a
1417 // displacement of q. Does not modify P.
1418 def STDPtrQRr : FSTDLDD<1,
1419                         (outs),
1420                         (ins memri:$memri, GPR8:$reg),
1421                         "std\t$memri, $reg",
1422                         [(store i8:$reg, addr:$memri)]>,
1423                 Requires<[HasSRAM]>;
1424
1425 // STDW P+q, Rr+1:Rr
1426 // Stores the value of Rr into the location addressed by pointer P with a
1427 // displacement of q. Does not modify P.
1428 //
1429 // Expands to:
1430 // std P+q,   Rr
1431 // std P+q+1, Rr+1
1432 def STDWPtrQRr : Pseudo<(outs),
1433                         (ins memri:$memri, DREGS:$src),
1434                         "stdw\t$memri, $src",
1435                         [(store i16:$src, addr:$memri)]>,
1436                  Requires<[HasSRAM]>;
1437
1438
1439 // Load program memory operations.
1440 let canFoldAsLoad = 1,
1441 isReMaterializable = 1,
1442 mayLoad = 1,
1443 hasSideEffects = 0 in
1444 {
1445   let Defs = [R0],
1446       Uses = [R31R30] in
1447   def LPM : F16<0b1001010111001000,
1448                 (outs),
1449                 (ins),
1450                 "lpm",
1451                 []>,
1452             Requires<[HasLPM]>;
1453
1454   def LPMRdZ : FLPMX<0,
1455                      0,
1456                      (outs GPR8:$dst),
1457                      (ins ZREG:$z),
1458                      "lpm\t$dst, $z",
1459                      []>,
1460                Requires<[HasLPMX]>;
1461
1462   // Load program memory, while postincrementing the Z register.
1463   let Defs = [R31R30] in
1464   {
1465     def LPMRdZPi : FLPMX<0,
1466                          1,
1467                          (outs GPR8:$dst),
1468                          (ins ZREG:$z),
1469                          "lpm\t$dst, $z+",
1470                          []>,
1471                    Requires<[HasLPMX]>;
1472
1473     def LPMWRdZ : Pseudo<(outs DREGS:$dst),
1474                          (ins ZREG:$z),
1475                          "lpmw\t$dst, $z",
1476                          []>,
1477                   Requires<[HasLPMX]>;
1478
1479     def LPMWRdZPi : Pseudo<(outs DREGS:$dst),
1480                            (ins ZREG:$z),
1481                            "lpmw\t$dst, $z+",
1482                            []>,
1483                     Requires<[HasLPMX]>;
1484   }
1485 }
1486
1487 // Extended load program memory operations.
1488 let mayLoad = 1,
1489 hasSideEffects = 0 in
1490 {
1491   let Defs = [R0],
1492       Uses = [R31R30] in
1493   def ELPM : F16<0b1001010111011000,
1494                  (outs),
1495                  (ins),
1496                  "elpm",
1497                  []>,
1498              Requires<[HasELPM]>;
1499
1500   def ELPMRdZ : FLPMX<1,
1501                       0,
1502                       (outs GPR8:$dst),
1503                       (ins ZREG:$z),
1504                       "elpm\t$dst, $z",
1505                       []>,
1506                 Requires<[HasELPMX]>;
1507
1508   let Defs = [R31R30] in
1509   def ELPMRdZPi : FLPMX<1,
1510                         1,
1511                         (outs GPR8:$dst),
1512                         (ins ZREG: $z),
1513                         "elpm\t$dst, $z+",
1514                         []>,
1515                   Requires<[HasELPMX]>;
1516 }
1517
1518 // Store program memory operations.
1519 let Uses = [R1, R0] in
1520 {
1521   let Uses = [R31R30, R1, R0] in
1522   def SPM : F16<0b1001010111101000,
1523                 (outs),
1524                 (ins),
1525                 "spm",
1526                 []>,
1527             Requires<[HasSPM]>;
1528
1529   let Defs = [R31R30] in
1530   def SPMZPi : F16<0b1001010111111000,
1531                    (outs),
1532                    (ins ZREG:$z),
1533                    "spm $z+",
1534                    []>,
1535                Requires<[HasSPMX]>;
1536 }
1537
1538 // Read data from IO location operations.
1539 let canFoldAsLoad = 1,
1540 isReMaterializable = 1 in
1541 {
1542   def INRdA : FIORdA<(outs GPR8:$dst),
1543                      (ins imm_port6:$src),
1544                      "in\t$dst, $src",
1545                      [(set i8:$dst, (load ioaddr8:$src))]>;
1546
1547   def INWRdA : Pseudo<(outs DREGS:$dst),
1548                       (ins imm_port6:$src),
1549                       "inw\t$dst, $src",
1550                       [(set i16:$dst, (load ioaddr16:$src))]>;
1551 }
1552
1553 // Write data to IO location operations.
1554 def OUTARr : FIOARr<(outs),
1555                     (ins imm_port6:$dst, GPR8:$src),
1556                     "out\t$dst, $src",
1557                     [(store i8:$src, ioaddr8:$dst)]>;
1558
1559 def OUTWARr : Pseudo<(outs),
1560                      (ins imm_port6:$dst, DREGS:$src),
1561                      "outw\t$dst, $src",
1562                      [(store i16:$src, ioaddr16:$dst)]>;
1563
1564 // Stack push/pop operations.
1565 let Defs = [SP],
1566 Uses = [SP],
1567 hasSideEffects = 0 in
1568 {
1569   // Stack push operations.
1570   let mayStore = 1 in
1571   {
1572     def PUSHRr : FRd<0b1001,
1573                      0b0011111,
1574                      (outs),
1575                      (ins GPR8:$reg),
1576                      "push\t$reg",
1577                      []>,
1578                  Requires<[HasSRAM]>;
1579
1580     def PUSHWRr : Pseudo<(outs),
1581                          (ins DREGS:$reg),
1582                          "pushw\t$reg",
1583                          []>,
1584                   Requires<[HasSRAM]>;
1585   }
1586
1587   // Stack pop operations.
1588   let mayLoad = 1 in
1589   {
1590     def POPRd : FRd<0b1001,
1591                     0b0001111,
1592                     (outs GPR8:$reg),
1593                     (ins),
1594                     "pop\t$reg",
1595                     []>,
1596                 Requires<[HasSRAM]>;
1597
1598     def POPWRd : Pseudo<(outs DREGS:$reg),
1599                         (ins),
1600                         "popw\t$reg",
1601                         []>,
1602                  Requires<[HasSRAM]>;
1603   }
1604 }
1605
1606 // Read-Write-Modify (RMW) instructions.
1607 def XCHZRd : FZRd<0b100,
1608                   (outs GPR8:$rd),
1609                   (ins ZREG:$z),
1610                   "xch\t$z, $rd",
1611                   []>,
1612              Requires<[SupportsRMW]>;
1613
1614 def LASZRd : FZRd<0b101,
1615                   (outs GPR8:$rd),
1616                   (ins ZREG:$z),
1617                   "las\t$z, $rd",
1618                   []>,
1619              Requires<[SupportsRMW]>;
1620
1621 def LACZRd : FZRd<0b110,
1622                   (outs GPR8:$rd),
1623                   (ins ZREG:$z),
1624                   "lac\t$z, $rd",
1625                   []>,
1626              Requires<[SupportsRMW]>;
1627
1628 def LATZRd : FZRd<0b111,
1629                   (outs GPR8:$rd),
1630                   (ins ZREG:$z),
1631                   "lat\t$z, $rd",
1632                   []>,
1633              Requires<[SupportsRMW]>;
1634
1635 //===----------------------------------------------------------------------===//
1636 // Bit and bit-test instructions
1637 //===----------------------------------------------------------------------===//
1638
1639 // Bit shift/rotate operations.
1640 let Constraints = "$src = $rd",
1641 Defs = [SREG] in
1642 {
1643   // 8-bit LSL is an alias of ADD Rd, Rd
1644
1645   def LSLWRd : Pseudo<(outs DREGS:$rd),
1646                       (ins DREGS:$src),
1647                       "lslw\t$rd",
1648                       [(set i16:$rd, (AVRlsl i16:$src)), (implicit SREG)]>;
1649
1650   def LSRRd : FRd<0b1001,
1651                   0b0100110,
1652                   (outs GPR8:$rd),
1653                   (ins GPR8:$src),
1654                   "lsr\t$rd",
1655                   [(set i8:$rd, (AVRlsr i8:$src)), (implicit SREG)]>;
1656
1657   def LSRWRd : Pseudo<(outs DREGS:$rd),
1658                       (ins DREGS:$src),
1659                       "lsrw\t$rd",
1660                       [(set i16:$rd, (AVRlsr i16:$src)), (implicit SREG)]>;
1661
1662   def ASRRd : FRd<0b1001,
1663                   0b0100101,
1664                   (outs GPR8:$rd),
1665                   (ins GPR8:$src),
1666                   "asr\t$rd",
1667                   [(set i8:$rd, (AVRasr i8:$src)), (implicit SREG)]>;
1668
1669   def ASRWRd : Pseudo<(outs DREGS:$rd),
1670                       (ins DREGS:$src),
1671                       "asrw\t$rd",
1672                       [(set i16:$rd, (AVRasr i16:$src)), (implicit SREG)]>;
1673
1674   // Bit rotate operations.
1675   let Uses = [SREG] in
1676   {
1677     // 8-bit ROL is an alias of ADC Rd, Rd
1678
1679     def ROLBRd : Pseudo<(outs GPR8:$rd),
1680                         (ins GPR8:$src),
1681                         "rolb\t$rd",
1682                         [(set i8:$rd, (AVRrol i8:$src)), (implicit SREG)]>;
1683
1684     def RORBRd : Pseudo<(outs GPR8:$rd),
1685                         (ins GPR8:$src),
1686                         "rorb\t$rd",
1687                         [(set i8:$rd, (AVRror i8:$src)), (implicit SREG)]>;
1688
1689     def ROLWRd : Pseudo<(outs DREGS:$rd),
1690                         (ins DREGS:$src),
1691                         "rolw\t$rd",
1692                         [(set i16:$rd, (AVRrol i16:$src)), (implicit SREG)]>;
1693
1694     def RORRd : FRd<0b1001,
1695                     0b0100111,
1696                     (outs GPR8:$rd),
1697                     (ins GPR8:$src),
1698                     "ror\t$rd",
1699                     []>;
1700
1701     def RORWRd : Pseudo<(outs DREGS:$rd),
1702                         (ins DREGS:$src),
1703                         "rorw\t$rd",
1704                         [(set i16:$rd, (AVRror i16:$src)), (implicit SREG)]>;
1705   }
1706 }
1707
1708 // SWAP Rd
1709 // Swaps the high and low nibbles in a register.
1710 let Constraints = "$src = $rd" in
1711 def SWAPRd : FRd<0b1001,
1712                  0b0100010,
1713                  (outs GPR8:$rd),
1714                  (ins GPR8:$src),
1715                  "swap\t$rd",
1716                  [(set i8:$rd, (bswap i8:$src))]>;
1717
1718 // IO register bit set/clear operations.
1719 //:TODO: add patterns when popcount(imm)==2 to be expanded with 2 sbi/cbi
1720 // instead of in+ori+out which requires one more instr.
1721 def SBIAb : FIOBIT<0b10,
1722                    (outs),
1723                    (ins imm_port5:$addr, i8imm:$bit),
1724                    "sbi\t$addr, $bit",
1725                    [(store (or (i8 (load lowioaddr8:$addr)), iobitpos8:$bit),
1726                      lowioaddr8:$addr)]>;
1727
1728 def CBIAb : FIOBIT<0b00,
1729                    (outs),
1730                    (ins imm_port5:$addr, i8imm:$bit),
1731                    "cbi\t$addr, $bit",
1732                    [(store (and (i8 (load lowioaddr8:$addr)), iobitposn8:$bit),
1733                      lowioaddr8:$addr)]>;
1734
1735 // Status register bit load/store operations.
1736 let Defs = [SREG] in
1737 def BST : FRdB<0b01,
1738                (outs),
1739                (ins GPR8:$rd, i8imm:$b),
1740                "bst\t$rd, $b",
1741                []>;
1742
1743 let Uses = [SREG] in
1744 def BLD : FRdB<0b00,
1745                (outs),
1746                (ins GPR8:$rd, i8imm:$b),
1747                "bld\t$rd, $b",
1748                []>;
1749
1750 def CBR : InstAlias<"cbr\t$rd, $k", (ANDIRdK LD8:$rd, imm_com8:$k), 0>;
1751
1752 // CLR Rd
1753 // Alias for EOR Rd, Rd
1754 // -------------
1755 // Clears all bits in a register.
1756 def CLR : InstAlias<"clr\t$rd", (EORRdRr GPR8:$rd, GPR8:$rd)>;
1757
1758 // LSL Rd
1759 // Alias for ADD Rd, Rd
1760 // --------------
1761 // Logical shift left one bit.
1762 def LSL : InstAlias<"lsl\t$rd", (ADDRdRr GPR8:$rd, GPR8:$rd)>;
1763
1764 def ROL : InstAlias<"rol\t$rd", (ADCRdRr GPR8:$rd, GPR8:$rd)>;
1765
1766 // SER Rd
1767 // Alias for LDI Rd, 0xff
1768 // ---------
1769 // Sets all bits in a register.
1770 def : InstAlias<"ser\t$rd", (LDIRdK LD8:$rd, 0xff), 0>;
1771
1772 let Defs = [SREG] in
1773 def BSETs : FS<0,
1774                (outs),
1775                (ins i8imm:$s),
1776                "bset\t$s",
1777                []>;
1778
1779 let Defs = [SREG] in
1780 def BCLRs : FS<1,
1781                (outs),
1782                (ins i8imm:$s),
1783                "bclr\t$s",
1784                []>;
1785
1786 // Set/clear aliases for the carry (C) status flag (bit 0).
1787 def : InstAlias<"sec", (BSETs 0)>;
1788 def : InstAlias<"clc", (BCLRs 0)>;
1789
1790 // Set/clear aliases for the zero (Z) status flag (bit 1).
1791 def : InstAlias<"sez", (BSETs 1)>;
1792 def : InstAlias<"clz", (BCLRs 1)>;
1793
1794 // Set/clear aliases for the negative (N) status flag (bit 2).
1795 def : InstAlias<"sen", (BSETs 2)>;
1796 def : InstAlias<"cln", (BCLRs 2)>;
1797
1798 // Set/clear aliases for the overflow (V) status flag (bit 3).
1799 def : InstAlias<"sev", (BSETs 3)>;
1800 def : InstAlias<"clv", (BCLRs 3)>;
1801
1802 // Set/clear aliases for the signed (S) status flag (bit 4).
1803 def : InstAlias<"ses", (BSETs 4)>;
1804 def : InstAlias<"cls", (BCLRs 4)>;
1805
1806 // Set/clear aliases for the half-carry (H) status flag (bit 5).
1807 def : InstAlias<"seh", (BSETs 5)>;
1808 def : InstAlias<"clh", (BCLRs 5)>;
1809
1810 // Set/clear aliases for the T status flag (bit 6).
1811 def : InstAlias<"set", (BSETs 6)>;
1812 def : InstAlias<"clt", (BCLRs 6)>;
1813
1814 // Set/clear aliases for the interrupt (I) status flag (bit 7).
1815 def : InstAlias<"sei", (BSETs 7)>;
1816 def : InstAlias<"cli", (BCLRs 7)>;
1817
1818 //===----------------------------------------------------------------------===//
1819 // Special/Control instructions
1820 //===----------------------------------------------------------------------===//
1821
1822 // BREAK
1823 // Breakpoint instruction
1824 // ---------
1825 // <|1001|0101|1001|1000>
1826 def BREAK : F16<0b1001010110011000,
1827                 (outs),
1828                 (ins),
1829                 "break",
1830                 []>,
1831             Requires<[HasBREAK]>;
1832
1833 // NOP
1834 // No-operation instruction
1835 // ---------
1836 // <|0000|0000|0000|0000>
1837 def NOP : F16<0b0000000000000000,
1838               (outs),
1839               (ins),
1840               "nop",
1841               []>;
1842
1843 // SLEEP
1844 // Sleep instruction
1845 // ---------
1846 // <|1001|0101|1000|1000>
1847 def SLEEP : F16<0b1001010110001000,
1848                 (outs),
1849                 (ins),
1850                 "sleep",
1851                 []>;
1852
1853 // WDR
1854 // Watchdog reset
1855 // ---------
1856 // <|1001|0101|1010|1000>
1857 def WDR : F16<0b1001010110101000,
1858               (outs),
1859               (ins),
1860               "wdr",
1861               []>;
1862
1863 //===----------------------------------------------------------------------===//
1864 // Pseudo instructions for later expansion
1865 //===----------------------------------------------------------------------===//
1866
1867 //:TODO: Optimize this for wider types AND optimize the following code
1868 //       compile int foo(char a, char b, char c, char d) {return d+b;}
1869 //       looks like a missed sext_inreg opportunity.
1870 def SEXT : ExtensionPseudo<
1871   (outs DREGS:$dst),
1872   (ins GPR8:$src),
1873   "sext\t$dst, $src",
1874   [(set i16:$dst, (sext i8:$src)), (implicit SREG)]
1875 >;
1876
1877 def ZEXT : ExtensionPseudo<
1878   (outs DREGS:$dst),
1879   (ins GPR8:$src),
1880   "zext\t$dst, $src",
1881   [(set i16:$dst, (zext i8:$src)), (implicit SREG)]
1882 >;
1883
1884 // This pseudo gets expanded into a movw+adiw thus it clobbers SREG.
1885 let Defs = [SREG],
1886     hasSideEffects = 0 in
1887 def FRMIDX : Pseudo<(outs DLDREGS:$dst),
1888                     (ins DLDREGS:$src, i16imm:$src2),
1889                     "frmidx\t$dst, $src, $src2",
1890                     []>;
1891
1892 // This pseudo is either converted to a regular store or a push which clobbers
1893 // SP.
1894 def STDSPQRr : StorePseudo<
1895   (outs),
1896   (ins memspi:$dst, GPR8:$src),
1897   "stdstk\t$dst, $src",
1898   [(store i8:$src, addr:$dst)]
1899 >;
1900
1901 // This pseudo is either converted to a regular store or a push which clobbers
1902 // SP.
1903 def STDWSPQRr : StorePseudo<
1904   (outs),
1905   (ins memspi:$dst, DREGS:$src),
1906   "stdwstk\t$dst, $src",
1907   [(store i16:$src, addr:$dst)]
1908 >;
1909
1910 // SP read/write pseudos.
1911 let hasSideEffects = 0 in
1912 {
1913   let Uses = [SP] in
1914   def SPREAD : Pseudo<
1915     (outs DREGS:$dst),
1916     (ins GPRSP:$src),
1917     "spread\t$dst, $src",
1918     []
1919   >;
1920
1921   let Defs = [SP] in
1922   def SPWRITE : Pseudo<
1923     (outs GPRSP:$dst),
1924     (ins DREGS:$src),
1925     "spwrite\t$dst, $src",
1926     []>;
1927 }
1928
1929 def Select8 : SelectPseudo<
1930   (outs GPR8:$dst),
1931   (ins GPR8:$src, GPR8:$src2, i8imm:$cc),
1932   "# Select8 PSEUDO",
1933   [(set i8:$dst, (AVRselectcc i8:$src, i8:$src2, imm:$cc))]
1934 >;
1935
1936 def Select16 : SelectPseudo<
1937   (outs DREGS:$dst),
1938   (ins DREGS:$src, DREGS:$src2, i8imm:$cc),
1939   "# Select16 PSEUDO",
1940   [(set i16:$dst, (AVRselectcc i16:$src, i16:$src2, imm:$cc))]
1941 >;
1942
1943 def Lsl8 : ShiftPseudo<
1944   (outs GPR8:$dst),
1945   (ins GPR8:$src, GPR8:$cnt),
1946   "# Lsl8 PSEUDO",
1947   [(set i8:$dst, (AVRlslLoop i8:$src, i8:$cnt))]
1948 >;
1949
1950 def Lsl16 : ShiftPseudo<
1951   (outs DREGS:$dst),
1952   (ins DREGS:$src, GPR8:$cnt),
1953   "# Lsl16 PSEUDO",
1954   [(set i16:$dst, (AVRlslLoop i16:$src, i8:$cnt))]
1955 >;
1956
1957 def Lsr8 : ShiftPseudo<
1958   (outs GPR8:$dst),
1959   (ins GPR8:$src, GPR8:$cnt),
1960   "# Lsr8 PSEUDO",
1961   [(set i8:$dst, (AVRlsrLoop i8:$src, i8:$cnt))]
1962 >;
1963
1964 def Lsr16 : ShiftPseudo<
1965   (outs DREGS:$dst),
1966    (ins DREGS:$src, GPR8:$cnt),
1967    "# Lsr16 PSEUDO",
1968    [(set i16:$dst, (AVRlsrLoop i16:$src, i8:$cnt))]
1969 >;
1970
1971 def Rol8 : ShiftPseudo<
1972   (outs GPR8:$dst),
1973   (ins GPR8:$src, GPR8:$cnt),
1974   "# Rol8 PSEUDO",
1975   [(set i8:$dst, (AVRrolLoop i8:$src, i8:$cnt))]
1976 >;
1977
1978 def Rol16 : ShiftPseudo<
1979   (outs DREGS:$dst),
1980   (ins DREGS:$src, GPR8:$cnt),
1981   "# Rol16 PSEUDO",
1982   [(set i16:$dst, (AVRrolLoop i16:$src, i8:$cnt))]
1983 >;
1984
1985 def Ror8 : ShiftPseudo<
1986   (outs GPR8:$dst),
1987   (ins GPR8:$src, GPR8:$cnt),
1988   "# Ror8 PSEUDO",
1989   [(set i8:$dst, (AVRrorLoop i8:$src, i8:$cnt))]
1990 >;
1991
1992 def Ror16 : ShiftPseudo<
1993   (outs DREGS:$dst),
1994   (ins DREGS:$src, GPR8:$cnt),
1995   "# Ror16 PSEUDO",
1996   [(set i16:$dst, (AVRrorLoop i16:$src, i8:$cnt))]
1997 >;
1998
1999 def Asr8 : ShiftPseudo<
2000   (outs GPR8:$dst),
2001   (ins GPR8:$src, GPR8:$cnt),
2002   "# Asr8 PSEUDO",
2003   [(set i8:$dst, (AVRasrLoop i8:$src, i8:$cnt))]
2004 >;
2005
2006 def Asr16 : ShiftPseudo<
2007   (outs DREGS:$dst),
2008    (ins DREGS:$src, GPR8:$cnt),
2009    "# Asr16 PSEUDO",
2010    [(set i16:$dst, (AVRasrLoop i16:$src, i8:$cnt))]
2011 >;
2012
2013
2014 //===----------------------------------------------------------------------===//
2015 // Non-Instruction Patterns
2016 //===----------------------------------------------------------------------===//
2017
2018 //:TODO: look in x86InstrCompiler.td for odd encoding trick related to
2019 // add x, 128 -> sub x, -128. Clang is emitting an eor for this (ldi+eor)
2020
2021 // the add instruction always writes the carry flag
2022 def : Pat<(addc i8:$src, i8:$src2),
2023           (ADDRdRr i8:$src, i8:$src2)>;
2024 def : Pat<(addc DREGS:$src, DREGS:$src2),
2025           (ADDWRdRr DREGS:$src, DREGS:$src2)>;
2026
2027 // all sub instruction variants always writes the carry flag
2028 def : Pat<(subc i8:$src, i8:$src2),
2029           (SUBRdRr i8:$src, i8:$src2)>;
2030 def : Pat<(subc i16:$src, i16:$src2),
2031           (SUBWRdRr i16:$src, i16:$src2)>;
2032 def : Pat<(subc i8:$src, imm:$src2),
2033           (SUBIRdK i8:$src, imm:$src2)>;
2034 def : Pat<(subc i16:$src, imm:$src2),
2035           (SUBIWRdK i16:$src, imm:$src2)>;
2036
2037 // These patterns convert add (x, -imm) to sub (x, imm) since we dont have
2038 // any add with imm instructions. Also take care of the adiw/sbiw instructions.
2039 def : Pat<(add i16:$src1, imm0_63_neg:$src2),
2040           (SBIWRdK i16:$src1, (imm0_63_neg:$src2))>;
2041 def : Pat<(add i16:$src1, imm:$src2),
2042           (SUBIWRdK i16:$src1, (imm16_neg_XFORM imm:$src2))>;
2043 def : Pat<(addc i16:$src1, imm:$src2),
2044           (SUBIWRdK i16:$src1, (imm16_neg_XFORM imm:$src2))>;
2045 def : Pat<(adde i16:$src1, imm:$src2),
2046           (SBCIWRdK i16:$src1, (imm16_neg_XFORM imm:$src2))>;
2047
2048 def : Pat<(add i8:$src1, imm:$src2),
2049           (SUBIRdK i8:$src1, (imm8_neg_XFORM imm:$src2))>;
2050 def : Pat<(addc i8:$src1, imm:$src2),
2051           (SUBIRdK i8:$src1, (imm8_neg_XFORM imm:$src2))>;
2052 def : Pat<(adde i8:$src1, imm:$src2),
2053           (SBCIRdK i8:$src1, (imm8_neg_XFORM imm:$src2))>;
2054
2055 // Calls.
2056 def : Pat<(AVRcall (i16 tglobaladdr:$dst)),
2057           (CALLk tglobaladdr:$dst)>;
2058 def : Pat<(AVRcall (i16 texternalsym:$dst)),
2059           (CALLk texternalsym:$dst)>;
2060
2061 // `anyext`
2062 def : Pat<(i16 (anyext i8:$src)),
2063           (INSERT_SUBREG (i16 (IMPLICIT_DEF)), i8:$src, sub_lo)>;
2064
2065 // `trunc`
2066 def : Pat<(i8 (trunc i16:$src)),
2067           (EXTRACT_SUBREG i16:$src, sub_lo)>;
2068
2069 // sext_inreg
2070 def : Pat<(sext_inreg i16:$src, i8),
2071           (SEXT (i8 (EXTRACT_SUBREG i16:$src, sub_lo)))>;
2072
2073 // GlobalAddress
2074 def : Pat<(i16 (AVRWrapper tglobaladdr:$dst)),
2075           (LDIWRdK tglobaladdr:$dst)>;
2076 def : Pat<(add i16:$src, (AVRWrapper tglobaladdr:$src2)),
2077           (SUBIWRdK i16:$src, tglobaladdr:$src2)>;
2078 def : Pat<(i8 (load (AVRWrapper tglobaladdr:$dst))),
2079           (LDSRdK tglobaladdr:$dst)>;
2080 def : Pat<(i16 (load (AVRWrapper tglobaladdr:$dst))),
2081           (LDSWRdK tglobaladdr:$dst)>;
2082 def : Pat<(store i8:$src, (i16 (AVRWrapper tglobaladdr:$dst))),
2083           (STSKRr tglobaladdr:$dst, i8:$src)>;
2084 def : Pat<(store i16:$src, (i16 (AVRWrapper tglobaladdr:$dst))),
2085           (STSWKRr tglobaladdr:$dst, i16:$src)>;
2086
2087 // BlockAddress
2088 def : Pat<(i16 (AVRWrapper tblockaddress:$dst)),
2089           (LDIWRdK tblockaddress:$dst)>;
2090
2091 // hi-reg truncation : trunc(int16 >> 8)
2092 //:FIXME: i think it's better to emit an extract subreg node in the DAG than
2093 // all this mess once we get optimal shift code
2094 // lol... I think so, too. [@agnat]
2095 def : Pat<(i8 (trunc (AVRlsr (AVRlsr (AVRlsr (AVRlsr (AVRlsr (AVRlsr (AVRlsr
2096                      (AVRlsr DREGS:$src)))))))))),
2097           (EXTRACT_SUBREG DREGS:$src, sub_hi)>;
2098
2099 // :FIXME: DAGCombiner produces an shl node after legalization from these seq:
2100 // BR_JT -> (mul x, 2) -> (shl x, 1)
2101 def : Pat<(shl i16:$src1, (i8 1)),
2102           (LSLWRd i16:$src1)>;
2103
2104 // Lowering of 'tst' node to 'TST' instruction.
2105 // TST is an alias of AND Rd, Rd.
2106 def : Pat<(AVRtst i8:$rd),
2107           (ANDRdRr GPR8:$rd, GPR8:$rd)>;
2108
2109 // Lowering of 'lsl' node to 'LSL' instruction.
2110 // LSL is an alias of 'ADD Rd, Rd'
2111 def : Pat<(AVRlsl i8:$rd),
2112           (ADDRdRr GPR8:$rd, GPR8:$rd)>;
2113