gem5 v24.1.0.1
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
vector.cc
Go to the documentation of this file.
1/*
2 * Copyright (c) 2022 PLCT Lab
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are
7 * met: redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer;
9 * redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution;
12 * neither the name of the copyright holders nor the names of its
13 * contributors may be used to endorse or promote products derived from
14 * this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
30
31#include <sstream>
32#include <string>
33
35#include "arch/riscv/isa.hh"
38#include "arch/riscv/utility.hh"
39#include "cpu/static_inst.hh"
40
41namespace gem5
42{
43
44namespace RiscvISA
45{
46
61float
62getVflmul(uint32_t vlmul_encoding)
63{
64 int vlmul = sext<3>(vlmul_encoding & 7);
65 float vflmul = vlmul >= 0 ? 1 << vlmul : 1.0 / (1 << -vlmul);
66 return vflmul;
67}
68
69uint32_t
70getVlmax(VTYPE vtype, uint32_t vlen)
71{
72 uint32_t sew = getSew(vtype.vsew);
73 // vlmax is defined in RVV 1.0 spec p12 chapter 3.4.2.
74 uint32_t vlmax = (vlen/sew) * getVflmul(vtype.vlmul);
75 return vlmax;
76}
77
78std::string
80{
81 std::stringstream ss;
82 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", ";
83 if (bit31 && bit30 == 0) {
84 ss << registerName(srcRegIdx(0)) << ", " << registerName(srcRegIdx(1));
85 } else if (bit31 && bit30) {
86 ss << uimm << ", " << generateZimmDisassembly();
87 } else {
89 }
90 return ss.str();
91}
92
93std::string
95{
96 std::stringstream s;
97
98 // VSETIVLI uses ZIMM10 and VSETVLI uses ZIMM11
99 uint64_t zimm = (bit31 && bit30) ? zimm10 : zimm11;
100
101 bool frac_lmul = bits(zimm, 2);
102 int sew = 1 << (bits(zimm, 5, 3) + 3);
103 int lmul = bits(zimm, 1, 0);
104 auto vta = bits(zimm, 6) == 1 ? "ta" : "tu";
105 auto vma = bits(zimm, 7) == 1 ? "ma" : "mu";
106 s << "e" << sew;
107 if (frac_lmul) {
108 std::string lmul_str = "";
109 switch(lmul){
110 case 3:
111 lmul_str = "f2";
112 break;
113 case 2:
114 lmul_str = "f4";
115 break;
116 case 1:
117 lmul_str = "f8";
118 break;
119 default:
120 panic("Unsupport fractional LMUL");
121 }
122 s << ", m" << lmul_str;
123 } else {
124 s << ", m" << (1 << lmul);
125 }
126 s << ", " << vta << ", " << vma;
127 return s.str();
128}
129
130std::string
132 const loader::SymbolTable *symtab) const
133{
134 std::stringstream ss;
135 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", "
136 << registerName(srcRegIdx(0));
137 if (machInst.vm == 0) ss << ", v0.t";
138 return ss.str();
139}
140
142 const loader::SymbolTable *symtab) const
143{
144 std::stringstream ss;
145 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", ";
146 if (machInst.funct3 == 0x3) {
147 // OPIVI
148 ss << registerName(srcRegIdx(0)) << ", " << machInst.vecimm;
149 } else {
150 ss << registerName(srcRegIdx(1)) << ", " << registerName(srcRegIdx(0));
151 }
152 if (machInst.vm == 0) ss << ", v0.t";
153 return ss.str();
154}
155
157 const loader::SymbolTable *symtab) const
158{
159 std::stringstream ss;
160 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", ";
161 if (machInst.funct3 == 0x3) {
162 // OPIVI
163 ss << registerName(srcRegIdx(0)) << ", " << machInst.vecimm;
164 } else {
165 ss << registerName(srcRegIdx(1)) << ", " << registerName(srcRegIdx(0));
166 }
167 if (machInst.vm == 0) ss << ", v0.t";
168 return ss.str();
169}
170
172 const loader::SymbolTable *symtab) const
173{
174 std::stringstream ss;
175 ss << mnemonic << ' ' << registerName(destRegIdx(0));
176 if (machInst.vm == 0) ss << ", v0.t";
177 return ss.str();
178}
179
181 const loader::SymbolTable *symtab) const
182{
183 std::stringstream ss;
184 ss << mnemonic << ' ' << registerName(destRegIdx(0));
185 if (machInst.vm == 0) ss << ", v0.t";
186 return ss.str();
187}
188
190 const loader::SymbolTable *symtab) const
191{
192 std::stringstream ss;
193 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", ";
194 if (machInst.funct3 == 0x3) {
195 ss << registerName(srcRegIdx(0)) << ", " << machInst.vecimm;
196 } else {
197 ss << registerName(srcRegIdx(1)) << ", " << registerName(srcRegIdx(0));
198 }
199 if (machInst.vm == 0) ss << ", v0.t";
200 return ss.str();
201}
202
204 const loader::SymbolTable *symtab) const
205{
206 std::stringstream ss;
207 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", ";
208 if (machInst.funct3 == 0x3) {
209 ss << registerName(srcRegIdx(0)) << ", " << machInst.vecimm;
210 } else {
211 ss << registerName(srcRegIdx(1)) << ", " << registerName(srcRegIdx(0));
212 }
213 if (machInst.vm == 0) ss << ", v0.t";
214 return ss.str();
215}
216
218 const loader::SymbolTable *symtab) const
219{
220 std::stringstream ss;
221 unsigned vlenb = vlen >> 3;
222 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", "
223 << vlenb * microIdx << '(' << registerName(srcRegIdx(0)) << ')' << ", "
224 << registerName(srcRegIdx(1));
225 if (!machInst.vm) ss << ", v0.t";
226 return ss.str();
227}
228
230 const loader::SymbolTable *symtab) const
231{
232 std::stringstream ss;
233 unsigned vlenb = vlen >> 3;
234 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", "
235 << vlenb * microIdx << '(' << registerName(srcRegIdx(0)) << ')';
236 return ss.str();
237}
238
240 const loader::SymbolTable *symtab) const
241{
242 std::stringstream ss;
243 unsigned vlenb = vlen >> 3;
244 ss << mnemonic << ' ' << registerName(srcRegIdx(1)) << ", "
245 << vlenb * microIdx << '(' << registerName(srcRegIdx(0)) << ')';
246 if (!machInst.vm) ss << ", v0.t";
247 return ss.str();
248}
249
251 const loader::SymbolTable *symtab) const
252{
253 std::stringstream ss;
254 unsigned vlenb = vlen >> 3;
255 ss << mnemonic << ' ' << registerName(srcRegIdx(1)) << ", "
256 << vlenb * microIdx << '(' << registerName(srcRegIdx(0)) << ')';
257 return ss.str();
258}
259
261 const loader::SymbolTable *symtab) const
262{
263 std::stringstream ss;
264 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
265 '(' << registerName(srcRegIdx(0)) << ')';
266 if (!machInst.vm) ss << ", v0.t";
267 return ss.str();
268}
269
271 const loader::SymbolTable *symtab) const
272{
273 std::stringstream ss;
274 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
275 '(' << registerName(srcRegIdx(0)) << ')';
276 return ss.str();
277}
278
280 const loader::SymbolTable *symtab) const
281{
282 std::stringstream ss;
283 ss << mnemonic << ' ' << registerName(srcRegIdx(1)) << ", " <<
284 '(' << registerName(srcRegIdx(0)) << ')';
285 if (!machInst.vm) ss << ", v0.t";
286 return ss.str();
287}
288
290 const loader::SymbolTable *symtab) const
291{
292 std::stringstream ss;
293 ss << mnemonic << ' ' << registerName(srcRegIdx(1)) << ", " <<
294 '(' << registerName(srcRegIdx(0)) << ')';
295 return ss.str();
296}
297
299 const loader::SymbolTable *symtab) const
300{
301 std::stringstream ss;
302 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
303 '(' << registerName(srcRegIdx(0)) << ')' <<
304 ", " << registerName(srcRegIdx(1));
305 if (!machInst.vm) ss << ", v0.t";
306 return ss.str();
307}
308
310 const loader::SymbolTable *symtab) const
311{
312 std::stringstream ss;
313 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
314 '(' << registerName(srcRegIdx(0)) << ')' <<
315 ", "<< registerName(srcRegIdx(1));
316 if (microIdx != 0 || machInst.vtype8.vma == 0 || machInst.vtype8.vta == 0)
317 ss << ", " << registerName(srcRegIdx(2));
318 if (!machInst.vm) ss << ", v0.t";
319 return ss.str();
320}
321
323 const loader::SymbolTable *symtab) const
324{
325 std::stringstream ss;
326 ss << mnemonic << ' ' << registerName(srcRegIdx(2)) << ", " <<
327 '(' << registerName(srcRegIdx(0)) << ')' <<
328 ", " << registerName(srcRegIdx(1));
329 if (!machInst.vm) ss << ", v0.t";
330 return ss.str();
331}
332
334 const loader::SymbolTable *symtab) const
335{
336 std::stringstream ss;
337 ss << mnemonic << ' ' << registerName(srcRegIdx(2)) << ", " <<
338 '(' << registerName(srcRegIdx(0)) << ')' <<
339 ", "<< registerName(srcRegIdx(1));
340 if (microIdx != 0 || machInst.vtype8.vma == 0 || machInst.vtype8.vta == 0)
341 ss << ", " << registerName(srcRegIdx(2));
342 if (!machInst.vm) ss << ", v0.t";
343 return ss.str();
344}
345
347 const loader::SymbolTable *symtab) const
348{
349 std::stringstream ss;
350 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", "
351 << '(' << registerName(srcRegIdx(0)) << "),"
352 << registerName(srcRegIdx(1));
353 if (!machInst.vm) ss << ", v0.t";
354 return ss.str();
355}
356
358 const loader::SymbolTable *symtab) const
359{
360 std::stringstream ss;
361 ss << mnemonic << ' '
362 << registerName(destRegIdx(0)) << "[" << uint16_t(vdElemIdx) << "], "
363 << '(' << registerName(srcRegIdx(0)) << "), "
364 << registerName(srcRegIdx(1)) << "[" << uint16_t(vs2ElemIdx) << "]";
365 if (microIdx != 0 || machInst.vtype8.vma == 0 || machInst.vtype8.vta == 0)
366 ss << ", " << registerName(srcRegIdx(2));
367 if (!machInst.vm) ss << ", v0.t";
368 return ss.str();
369}
370
372 const loader::SymbolTable *symtab) const
373{
374 std::stringstream ss;
375 ss << mnemonic << ' ' << registerName(srcRegIdx(2)) << ", "
376 << '(' << registerName(srcRegIdx(0)) << "),"
377 << registerName(srcRegIdx(1));
378 if (!machInst.vm) ss << ", v0.t";
379 return ss.str();
380}
381
383 const loader::SymbolTable *symtab) const
384{
385 std::stringstream ss;
386 ss << mnemonic << ' '
387 << registerName(srcRegIdx(2)) << "[" << uint16_t(vs3ElemIdx) << "], "
388 << '(' << registerName(srcRegIdx(0)) << "), "
389 << registerName(srcRegIdx(1)) << "[" << uint16_t(vs2ElemIdx) << "]";
390 if (!machInst.vm) ss << ", v0.t";
391 return ss.str();
392}
393
394std::string
396 const loader::SymbolTable *symtab) const
397{
398 std::stringstream ss;
399 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
401 return ss.str();
402}
403
404std::string
406 const loader::SymbolTable *symtab) const
407{
408 std::stringstream ss;
409 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
411 return ss.str();
412}
413
415 uint8_t _dstReg, uint8_t _numSrcs, uint32_t _elen, uint32_t _vlen,
416 size_t _elemSize)
417 : VectorArithMicroInst("vmask_mv_micro", extMachInst,
418 SimdAddOp, 0, 0, _elen, _vlen),
419 elemSize(_elemSize)
420{
422 reinterpret_cast<RegIdArrayPtr>(
423 &std::remove_pointer_t<decltype(this)>::srcRegIdxArr),
424 reinterpret_cast<RegIdArrayPtr>(
425 &std::remove_pointer_t<decltype(this)>::destRegIdxArr));
426
427 _numSrcRegs = 0;
428 _numDestRegs = 0;
429
432 for (uint8_t i=0; i<_numSrcs; i++) {
434 }
435}
436
437Fault
439 trace::InstRecord* traceData) const
440{
441 vreg_t& tmp_d0 = *(vreg_t *)xc->getWritableRegOperand(this, 0);
442 auto Vd = tmp_d0.as<uint8_t>();
443 uint32_t vlenb = vlen >> 3;
444 const uint32_t elems_per_vreg = vlenb / elemSize;
445 size_t bit_cnt = elems_per_vreg;
446
447 // mask tails are always treated as agnostic: writting 1s
448 tmp_d0.set(0xff);
449
450 vreg_t tmp_s;
451 for (uint8_t i = 0; i < this->_numSrcRegs; i++) {
452 xc->getRegOperand(this, i, &tmp_s);
453 auto s = tmp_s.as<uint8_t>();
454 if (elems_per_vreg < 8) {
455 const uint32_t m = (1 << elems_per_vreg) - 1;
456 const uint32_t mask = m << (i * elems_per_vreg % 8);
457 // clr & ext bits
458 Vd[bit_cnt/8] ^= Vd[bit_cnt/8] & mask;
459 Vd[bit_cnt/8] |= s[bit_cnt/8] & mask;
460 bit_cnt += elems_per_vreg;
461 } else {
462 const uint32_t byte_offset = elems_per_vreg / 8;
463 memcpy(Vd + i * byte_offset, s + i * byte_offset, byte_offset);
464 }
465 }
466 if (traceData) {
467 traceData->setData(vecRegClass, &tmp_d0);
468 }
469 return NoFault;
470}
471
472std::string
474 const loader::SymbolTable *symtab) const
475{
476 std::stringstream ss;
477 ss << mnemonic << ' ' << registerName(destRegIdx(0));
478 for (uint8_t i = 0; i < this->_numSrcRegs; i++) {
479 ss << ", " << registerName(srcRegIdx(i));
480 }
481 unsigned vlenb = vlen >> 3;
482 ss << ", offset:" << vlenb / elemSize;
483 return ss.str();
484}
485
486Fault
488{
490 auto vcsr = xc->readMiscReg(MISCREG_VCSR);
491 xc->setMiscReg(MISCREG_VCSR, ((vcsr&~1)|*vxsat));
492 return NoFault;
493}
494
495std::string
497 const loader::SymbolTable *symtab) const
498{
499 std::stringstream ss;
500 ss << mnemonic << ' ' << "VXSAT" << ", " << (*vxsat ? "0x1" : "0x0");
501 return ss.str();
502}
503
505 uint32_t _microIdx, uint32_t _elen, uint32_t _vlen,
507 : VectorMicroInst("vlff_trimvl_v_micro", _machInst, SimdConfigOp,
508 _microVl, _microIdx, _elen, _vlen),
509 microops(_microops)
510{
512 reinterpret_cast<RegIdArrayPtr>(
513 &std::remove_pointer_t<decltype(this)>::srcRegIdxArr),
514 nullptr
515 );
516
517 // Create data dependency with load micros
518 for (uint8_t i=0; i<microIdx; i++) {
519 setSrcRegIdx(_numSrcRegs++, vecRegClass[_machInst.vd + i]);
520 }
521
522 this->flags[IsControl] = true;
523 this->flags[IsIndirectControl] = true;
524 this->flags[IsInteger] = true;
525 this->flags[IsUncondControl] = true;
526}
527
528uint32_t
530{
531 uint32_t vl = 0;
532 for (uint8_t i=0; i<microIdx; i++) {
533 VleMicroInst& micro = static_cast<VleMicroInst&>(*microops[i]);
534 vl += micro.faultIdx;
535
536 if (micro.trimVl)
537 break;
538 }
539 return vl;
540}
541
542Fault
544{
545 auto tc = xc->tcBase();
546 MISA misa = xc->readMiscReg(MISCREG_ISA);
547 STATUS status = xc->readMiscReg(MISCREG_STATUS);
548 if (!misa.rvv || status.vs == VPUStatus::OFF) {
549 return std::make_shared<IllegalInstFault>(
550 "RVV is disabled or VPU is off", machInst);
551 }
552
553 PCState pc;
554 set(pc, xc->pcState());
555
556 uint32_t new_vl = calcVl();
557
558 tc->setMiscReg(MISCREG_VSTART, 0);
559
560 RegVal final_val = new_vl;
561 if (traceData) {
562 traceData->setData(miscRegClass, final_val);
563 }
564
565 pc.vl(new_vl);
566 xc->pcState(pc);
567
568 return NoFault;
569}
570
571std::unique_ptr<PCStateBase>
573{
574 PCStateBase *pc_ptr = tc->pcState().clone();
575
576 uint32_t new_vl = calcVl();
577
578 pc_ptr->as<PCState>().vl(new_vl);
579 return std::unique_ptr<PCStateBase>{pc_ptr};
580}
581
582std::string
584 const loader::SymbolTable *symtab) const
585{
586 std::stringstream ss;
587 ss << mnemonic << " vl";
588 return ss.str();
589}
590
592 const loader::SymbolTable *symtab) const
593{
594 std::stringstream ss;
595 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
596 '(' << registerName(srcRegIdx(0)) << ')' <<
597 ", " << registerName(srcRegIdx(1));
598 if (!machInst.vm)
599 ss << ", v0.t";
600 return ss.str();
601}
602
604 const loader::SymbolTable *symtab) const
605{
606 std::stringstream ss;
607 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
608 '(' << registerName(srcRegIdx(0)) << ')' <<
609 ", "<< registerName(srcRegIdx(1));
610 if (microIdx != 0 || machInst.vtype8.vma == 0 || machInst.vtype8.vta == 0)
611 ss << ", " << registerName(srcRegIdx(2));
612 if (!machInst.vm)
613 ss << ", v0.t";
614 return ss.str();
615}
616
618 uint32_t _micro_vl, uint32_t _dstReg,
619 uint32_t _numSrcs, uint32_t _microIdx,
620 uint32_t _numMicroops, uint32_t _field, uint32_t _elen,
621 uint32_t _vlen, uint32_t _sizeOfElement)
622 : VectorArithMicroInst("vlseg_deintrlv_micro", extMachInst,
623 SimdAddOp, 0, 0, _elen, _vlen)
624{
626 reinterpret_cast<RegIdArrayPtr>(
627 &std::remove_pointer_t<decltype(this)>::srcRegIdxArr),
628 reinterpret_cast<RegIdArrayPtr>(
629 &std::remove_pointer_t<decltype(this)>::destRegIdxArr));
630
631 _numSrcRegs = 0;
632 _numDestRegs = 0;
633 numSrcs = _numSrcs;
634 numMicroops = _numMicroops;
635 field =_field;
636 sizeOfElement = _sizeOfElement;
637 microIdx = _microIdx;
638 micro_vl = _micro_vl;
639
642 for (uint32_t i=0; i < _numSrcs; i++) {
643 uint32_t index = VecMemInternalReg0 + i + (microIdx * _numSrcs);
645 }
646}
647
648Fault
650{
651 vreg_t& tmp_d0 = *(vreg_t *)xc->getWritableRegOperand(this, 0);
652 auto Vd = tmp_d0.as<uint8_t>();
653 const uint32_t elems_per_vreg = micro_vl;
654 vreg_t tmp_s;
655 auto s = tmp_s.as<uint8_t>();
656 uint32_t elem = 0;
657 uint32_t index = field;
658 for (uint32_t i = 0; i < numSrcs; i++) {
659 xc->getRegOperand(this, i, &tmp_s);
660 s = tmp_s.as<uint8_t>();
661
662 // copy tail/inactive elements from vtmp src
663 if (i == field) {
664 tmp_d0 = tmp_s;
665 }
666
667 while (index < (i + 1) * elems_per_vreg)
668 {
669 memcpy(Vd + (elem * sizeOfElement),
670 s + ((index % elems_per_vreg) * sizeOfElement),
672 index += numSrcs;
673 elem++;
674 }
675 }
676 if (traceData) {
677 traceData->setData(vecRegClass, &tmp_d0);
678 }
679 return NoFault;
680}
681
682std::string
684 const
685{
686 std::stringstream ss;
687 ss << mnemonic << ' ' << registerName(destRegIdx(0));
688 for (uint8_t i = 0; i < this->_numSrcRegs; i++) {
689 ss << ", " << registerName(srcRegIdx(i));
690 }
691 ss << ", field: " << field;
692 return ss.str();
693}
694
696 const loader::SymbolTable *symtab) const
697{
698 std::stringstream ss;
699 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
700 '(' << registerName(srcRegIdx(0)) << ')' <<
701 ", " << registerName(srcRegIdx(1));
702 if (!machInst.vm)
703 ss << ", v0.t";
704 return ss.str();
705}
706
708 const loader::SymbolTable *symtab) const
709{
710 std::stringstream ss;
711 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", " <<
712 '(' << registerName(srcRegIdx(0)) << ')' <<
713 ", "<< registerName(srcRegIdx(1));
714 if (!machInst.vm)
715 ss << ", v0.t";
716 return ss.str();
717}
718
720 uint32_t _micro_vl, uint32_t _dstReg,
721 uint32_t _numSrcs, uint32_t _microIdx,
722 uint32_t _numMicroops, uint32_t _field, uint32_t _elen,
723 uint32_t _vlen, uint32_t _sizeOfElement)
724 : VectorArithMicroInst("vsseg_reintrlv_micro", extMachInst,
725 SimdAddOp, 0, 0, _elen, _vlen)
726{
728 reinterpret_cast<RegIdArrayPtr>(
729 &std::remove_pointer_t<decltype(this)>::srcRegIdxArr),
730 reinterpret_cast<RegIdArrayPtr>(
731 &std::remove_pointer_t<decltype(this)>::destRegIdxArr));
732
733 _numSrcRegs = 0;
734 _numDestRegs = 0;
735 numSrcs = _numSrcs;
736 numMicroops = _numMicroops;
737 field =_field;
738 sizeOfElement = _sizeOfElement;
739 microIdx = _microIdx;
740 micro_vl = _micro_vl;
741
743 (_microIdx * numSrcs)]);
744
746 for (uint8_t i=0; i<_numSrcs; i++) {
748 (microIdx)]);
749 }
750}
751
752Fault
754 trace::InstRecord* traceData) const
755{
756 const uint32_t elems_per_vreg = micro_vl;
757 vreg_t& tmp_d0 = *(vreg_t *)xc->getWritableRegOperand(this, 0);
758 auto Vd = tmp_d0.as<uint8_t>();
759
760 vreg_t tmp_s;
761 auto s = tmp_s.as<uint8_t>();
762 xc->getRegOperand(this, 0, &tmp_s);
763 s = tmp_s.as<uint8_t>();
764
765 uint32_t indexVd = 0;
766 uint32_t srcReg = (field * elems_per_vreg) % numSrcs;
767 uint32_t indexs = (field * elems_per_vreg) / numSrcs;
768
769 while (indexVd < elems_per_vreg) {
770 xc->getRegOperand(this, srcReg, &tmp_s);
771 s = tmp_s.as<uint8_t>();
772
773 memcpy(Vd + (indexVd * sizeOfElement),
774 s + (indexs * sizeOfElement),
776
777 indexVd++;
778 srcReg++;
779 if (srcReg >= numSrcs) {
780 srcReg = 0;
781 indexs++;
782 }
783 }
784
785 if (traceData) {
786 traceData->setData(vecRegClass, &tmp_d0);
787 }
788 return NoFault;
789}
790
791std::string
793 const loader::SymbolTable *symtab) const
794{
795 std::stringstream ss;
796 ss << mnemonic << ' ' << registerName(destRegIdx(0));
797 for (uint8_t i = 0; i < this->_numSrcRegs; i++) {
798 ss << ", " << registerName(srcRegIdx(i));
799 }
800 ss << ", field: " << field;
801 return ss.str();
802}
803
804VCpyVsMicroInst::VCpyVsMicroInst(ExtMachInst _machInst, uint32_t _microIdx,
805 uint8_t _vsRegIdx, uint32_t _elen,
806 uint32_t _vlen)
807 : VectorArithMicroInst("vcpyvs_v_micro", _machInst, SimdMiscOp, 0,
808 _microIdx, _elen, _vlen)
809{
811 reinterpret_cast<RegIdArrayPtr>(
812 &std::remove_pointer_t<decltype(this)>::srcRegIdxArr),
813 reinterpret_cast<RegIdArrayPtr>(
814 &std::remove_pointer_t<decltype(this)>::destRegIdxArr));
815
816 _numSrcRegs = 0;
817 _numDestRegs = 0;
820 setSrcRegIdx(_numSrcRegs++, vecRegClass[_vsRegIdx + _microIdx]);
821}
822
823Fault
825{
826 MISA misa = xc->readMiscReg(MISCREG_ISA);
827 STATUS status = xc->readMiscReg(MISCREG_STATUS);
828
829 if (!misa.rvv || status.vs == VPUStatus::OFF) {
830 return std::make_shared<IllegalInstFault>(
831 "RVV is disabled or VPU is off", machInst);
832 }
833
836
837 // copy vector source reg to vtmp
838 vreg_t& vtmp = *(vreg_t *)xc->getWritableRegOperand(this, 0);
839 vreg_t vs;
840 xc->getRegOperand(this, 0, &vs);
841 vtmp = vs;
842
843 if (traceData) {
844 traceData->setData(vecRegClass, &vtmp);
845 }
846
847 return NoFault;
848}
849
850std::string
852 const loader::SymbolTable *symtab) const
853{
854 std::stringstream ss;
855 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", "
856 << registerName(srcRegIdx(0));
857 return ss.str();
858}
859
860VPinVdMicroInst::VPinVdMicroInst(ExtMachInst _machInst, uint32_t _microIdx,
861 uint32_t _numVdPins, uint32_t _elen,
862 uint32_t _vlen, bool _hasVdOffset)
863 : VectorArithMicroInst("vpinvd_v_micro", _machInst, SimdMiscOp, 0,
864 _microIdx, _elen, _vlen)
865 , hasVdOffset(_hasVdOffset)
866{
868 reinterpret_cast<RegIdArrayPtr>(
869 &std::remove_pointer_t<decltype(this)>::srcRegIdxArr),
870 reinterpret_cast<RegIdArrayPtr>(
871 &std::remove_pointer_t<decltype(this)>::destRegIdxArr));
872
873 _numSrcRegs = 0;
874 _numDestRegs = 0;
875 setDestRegIdx(_numDestRegs++, vecRegClass[_machInst.vd + _microIdx]);
877 if (!_machInst.vtype8.vta || (!_machInst.vm && !_machInst.vtype8.vma)
878 || hasVdOffset) {
879 setSrcRegIdx(_numSrcRegs++, vecRegClass[_machInst.vd + _microIdx]);
880 }
881 RegId Vd = destRegIdx(0);
882 Vd.setNumPinnedWrites(_numVdPins);
883 setDestRegIdx(0, Vd);
884}
885
886Fault
888{
889 MISA misa = xc->readMiscReg(MISCREG_ISA);
890 STATUS status = xc->readMiscReg(MISCREG_STATUS);
891
892 if (!misa.rvv || status.vs == VPUStatus::OFF) {
893 return std::make_shared<IllegalInstFault>(
894 "RVV is disabled or VPU is off", machInst);
895 }
896
899
900 // tail/mask policy: both undisturbed if one is, 1s if none
901 vreg_t& vd = *(vreg_t *)xc->getWritableRegOperand(this, 0);
902 if (!machInst.vtype8.vta || (!machInst.vm && !machInst.vtype8.vma)
903 || hasVdOffset) {
904 vreg_t old_vd;
905 xc->getRegOperand(this, 0, &old_vd);
906 vd = old_vd;
907 } else {
908 vd.set(0xff);
909 }
910
911 if (traceData) {
912 traceData->setData(vecRegClass, &vd);
913 }
914
915 return NoFault;
916}
917
918std::string
920 const loader::SymbolTable *symtab) const
921{
922 std::stringstream ss;
923 ss << mnemonic << ' ' << registerName(destRegIdx(0)) << ", ";
924
925 if (!machInst.vtype8.vta || (!machInst.vm && !machInst.vtype8.vma)
926 || hasVdOffset) {
928 } else {
929 ss << "~0";
930 }
931
932 return ss.str();
933}
934
935} // namespace RiscvISA
936} // namespace gem5
The ExecContext is an abstract base class the provides the interface used by the ISA to manipulate th...
virtual void * getWritableRegOperand(const StaticInst *si, int idx)=0
virtual ThreadContext * tcBase() const =0
Returns a pointer to the ThreadContext.
virtual RegVal readMiscReg(int misc_reg)=0
Reads a miscellaneous register, handling any architectural side effects due to reading that register.
virtual void setMiscReg(int misc_reg, RegVal val)=0
Sets a miscellaneous register, handling any architectural side effects due to writing that register.
virtual const PCStateBase & pcState() const =0
virtual RegVal getRegOperand(const StaticInst *si, int idx)=0
Target & as()
Definition pcstate.hh:73
virtual PCStateBase * clone() const =0
Register ID: describe an architectural register with its class and index.
Definition reg_class.hh:94
void setNumPinnedWrites(int num_writes)
Definition reg_class.hh:163
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:79
std::string generateZimmDisassembly() const
Definition vector.cc:94
VCpyVsMicroInst(ExtMachInst _machInst, uint32_t _microIdx, uint8_t _vsRegIdx, uint32_t _elen, uint32_t _vlen)
Definition vector.cc:804
Fault execute(ExecContext *, trace::InstRecord *) const override
Definition vector.cc:824
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:851
std::string generateDisassembly(Addr, const loader::SymbolTable *) const override
Internal function to generate disassembly string.
Definition vector.cc:473
VMaskMergeMicroInst(ExtMachInst extMachInst, uint8_t _dstReg, uint8_t _numSrcs, uint32_t _elen, uint32_t _vlen, size_t _elemSize)
Definition vector.cc:414
Fault execute(ExecContext *, trace::InstRecord *) const override
Definition vector.cc:438
RegId srcRegIdxArr[NumVecInternalRegs]
Definition vector.hh:576
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:395
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:405
Fault execute(ExecContext *, trace::InstRecord *) const override
Definition vector.cc:887
VPinVdMicroInst(ExtMachInst _machInst, uint32_t _microIdx, uint32_t _numVdPins, uint32_t _elen, uint32_t _vlen, bool _hasVdOffset=false)
Definition vector.cc:860
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:919
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:156
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:141
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:131
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:203
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:189
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:180
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:171
Fault execute(ExecContext *, trace::InstRecord *) const override
Definition vector.cc:543
std::string generateDisassembly(Addr, const loader::SymbolTable *) const override
Internal function to generate disassembly string.
Definition vector.cc:583
VlFFTrimVlMicroOp(ExtMachInst _machInst, uint32_t _microVl, uint32_t _microIdx, uint32_t _elen, uint32_t _vlen, std::vector< StaticInstPtr > &_microops)
Definition vector.cc:504
std::unique_ptr< PCStateBase > branchTarget(ThreadContext *) const override
Return the target address for an indirect branch (jump).
Definition vector.cc:572
std::vector< StaticInstPtr > & microops
Definition vector.hh:611
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:346
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:357
Fault execute(ExecContext *, trace::InstRecord *) const override
Definition vector.cc:649
RegId srcRegIdxArr[NumVecInternalRegs]
Definition vector.hh:660
std::string generateDisassembly(Addr, const loader::SymbolTable *) const override
Internal function to generate disassembly string.
Definition vector.cc:683
VlSegDeIntrlvMicroInst(ExtMachInst extMachInst, uint32_t _micro_vl, uint32_t _dstReg, uint32_t _numSrcs, uint32_t _microIdx, uint32_t _numMicroops, uint32_t _field, uint32_t _elen, uint32_t _vlen, uint32_t _sizeOfElement)
Definition vector.cc:617
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:591
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:603
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:298
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:309
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:270
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:229
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:260
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:217
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:371
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:382
VsSegIntrlvMicroInst(ExtMachInst extMachInst, uint32_t _micro_vl, uint32_t _dstReg, uint32_t _numSrcs, uint32_t _microIdx, uint32_t _numMicroops, uint32_t _field, uint32_t _elen, uint32_t _vlen, uint32_t _sizeOfElement)
Definition vector.cc:719
std::string generateDisassembly(Addr, const loader::SymbolTable *) const override
Internal function to generate disassembly string.
Definition vector.cc:792
RegId srcRegIdxArr[NumVecInternalRegs]
Definition vector.hh:717
Fault execute(ExecContext *, trace::InstRecord *) const override
Definition vector.cc:753
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:695
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:707
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:322
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:333
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:289
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:250
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:279
std::string generateDisassembly(Addr pc, const loader::SymbolTable *symtab) const override
Internal function to generate disassembly string.
Definition vector.cc:239
Fault execute(ExecContext *, trace::InstRecord *) const override
Definition vector.cc:487
std::string generateDisassembly(Addr, const loader::SymbolTable *) const override
Internal function to generate disassembly string.
Definition vector.cc:496
void setDestRegIdx(int i, const RegId &val)
void setRegIdxArrays(RegIdArrayPtr src, RegIdArrayPtr dest)
Set the pointers which point to the arrays of source and destination register indices.
std::array< uint8_t, MiscRegClass+1 > _numTypedDestRegs
RegId(StaticInst::*)[] RegIdArrayPtr
uint8_t _numSrcRegs
See numSrcRegs().
uint8_t _numDestRegs
See numDestRegs().
const RegId & destRegIdx(int i) const
Return logical index (architectural reg num) of i'th destination reg.
void setSrcRegIdx(int i, const RegId &val)
const char * mnemonic
Base mnemonic (e.g., "add").
std::bitset< Num_Flags > flags
Flag values for this instruction.
const RegId & srcRegIdx(int i) const
Return logical index (architectural reg num) of i'th source reg.
ThreadContext is the external interface to all thread state for anything outside of the CPU.
virtual const PCStateBase & pcState() const =0
Vector Register Abstraction This generic class is the model in a particularization of MVC,...
Definition vec_reg.hh:126
void set(uint8_t val)
Set the container.
Definition vec_reg.hh:144
VecElem * as()
View interposers.
Definition vec_reg.hh:194
void setData(std::array< T, N > d)
STL vector class.
Definition stl.hh:37
constexpr T bits(T val, unsigned first, unsigned last)
Extract the bitfield from position 'first' to 'last' (inclusive) from 'val' and right justify it.
Definition bitfield.hh:79
#define panic(...)
This implements a cprintf based panic() function.
Definition logging.hh:188
Bitfield< 5, 0 > status
constexpr RegClass miscRegClass
Definition misc.hh:2975
Bitfield< 12, 11 > set
Bitfield< 0 > m
float getVflmul(uint32_t vlmul_encoding)
This function translates the 3-bit value of vlmul bits to the corresponding lmul value as specified i...
Definition vector.cc:62
Bitfield< 7 > vma
Definition vector.hh:79
const int VecMemInternalReg0
Definition vector.hh:67
Bitfield< 6 > vta
Definition vector.hh:80
Bitfield< 30, 0 > index
uint32_t getVlmax(VTYPE vtype, uint32_t vlen)
Definition vector.cc:70
Bitfield< 11, 7 > vd
Definition types.hh:168
std::string registerName(RegId reg)
Definition utility.hh:130
Bitfield< 2 > i
Bitfield< 57, 41 > vl
Definition types.hh:63
Bitfield< 2 > s
Bitfield< 2, 0 > vlmul
Definition vector.hh:82
constexpr RegClass vecRegClass
Definition vector.hh:71
Bitfield< 11, 8 > ss
@ MISCREG_STATUS
Definition misc.hh:76
Bitfield< 9, 5 > vs
Bitfield< 4 > pc
uint32_t getSew(uint32_t vsew)
Definition vector.hh:52
Copyright (c) 2024 Arm Limited All rights reserved.
Definition binary32.hh:36
std::shared_ptr< FaultBase > Fault
Definition types.hh:249
static const OpClass SimdAddOp
Definition op_class.hh:67
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition types.hh:147
static const OpClass SimdMiscOp
Definition op_class.hh:72
static const OpClass SimdConfigOp
Definition op_class.hh:134
uint64_t RegVal
Definition types.hh:173
constexpr decltype(nullptr) NoFault
Definition types.hh:253
@ VecRegClass
Vector Register.
Definition reg_class.hh:64

Generated on Mon Jan 13 2025 04:28:28 for gem5 by doxygen 1.9.8