2 * Copyright (c) 2004-2005 The Regents of The University of Michigan
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.
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.
30 * Emulation of the Tsunami CChip CSRs
37 #include "arch/alpha/ev5.hh"
38 #include "base/trace.hh"
39 #include "dev/tsunami_cchip.hh"
40 #include "dev/tsunamireg.h"
41 #include "dev/tsunami.hh"
42 #include "mem/port.hh"
43 #include "cpu/exec_context.hh"
44 #include "cpu/intr_control.hh"
45 #include "sim/builder.hh"
46 #include "sim/system.hh"
49 //Should this be AlphaISA?
50 using namespace TheISA
;
52 TsunamiCChip::TsunamiCChip(Params
*p
)
53 : BasicPioDevice(p
), tsunami(p
->tsunami
)
61 for (int x
= 0; x
< Tsunami::Max_CPUs
; x
++)
67 //Put back pointer in tsunami
68 tsunami
->cchip
= this;
72 TsunamiCChip::read(Packet
&pkt
)
74 DPRINTF(Tsunami
, "read va=%#x size=%d\n", pkt
.addr
, pkt
.size
);
76 assert(pkt
.result
== Unknown
);
77 assert(pkt
.addr
>= pioAddr
&& pkt
.addr
< pioAddr
+ pioSize
);
79 pkt
.time
= curTick
+ pioDelay
;
80 Addr regnum
= (pkt
.addr
- pioAddr
) >> 6;
81 Addr daddr
= (pkt
.addr
- pioAddr
);
87 case sizeof(uint64_t):
89 data64
= new uint64_t;
90 pkt
.data
= (uint8_t*)data64
;
92 data64
= (uint64_t*)pkt
.data
;
94 if (daddr
& TSDEV_CC_BDIMS
)
96 *data64
= dim
[(daddr
>> 4) & 0x3F];
100 if (daddr
& TSDEV_CC_BDIRS
)
102 *data64
= dir
[(daddr
>> 4) & 0x3F];
111 panic("TSDEV_CC_MTR not implemeted\n");
114 *data64
= (ipint
<< 8) & 0xF | (itint
<< 4) & 0xF |
115 (pkt
.req
->getCpuNum() & 0x3);
151 panic("TSDEV_CC_PRBEN not implemented\n");
157 panic("TSDEV_CC_IICx not implemented\n");
163 panic("TSDEV_CC_MPRx not implemented\n");
172 panic("default in cchip read reached, accessing 0x%x\n");
176 case sizeof(uint32_t):
177 case sizeof(uint16_t):
178 case sizeof(uint8_t):
180 panic("invalid access size(?) for tsunami register!\n");
182 DPRINTF(Tsunami
, "Tsunami CChip: read regnum=%#x size=%d data=%lld\n",
183 regnum
, pkt
.size
, *data64
);
185 pkt
.result
= Success
;
190 TsunamiCChip::write(Packet
&pkt
)
192 pkt
.time
= curTick
+ pioDelay
;
195 assert(pkt
.addr
>= pioAddr
&& pkt
.addr
< pioAddr
+ pioSize
);
196 Addr daddr
= pkt
.addr
- pioAddr
;
197 Addr regnum
= (pkt
.addr
- pioAddr
) >> 6 ;
200 uint64_t val
= *(uint64_t *)pkt
.data
;
201 assert(pkt
.size
== sizeof(uint64_t));
203 DPRINTF(Tsunami
, "write - addr=%#x value=%#x\n", pkt
.addr
, val
);
205 bool supportedWrite
= false;
208 if (daddr
& TSDEV_CC_BDIMS
)
210 int number
= (daddr
>> 4) & 0x3F;
216 olddim
= dim
[number
];
217 olddir
= dir
[number
];
219 dir
[number
] = dim
[number
] & drir
;
220 for(int x
= 0; x
< Tsunami::Max_CPUs
; x
++)
222 bitvector
= ULL(1) << x
;
223 // Figure out which bits have changed
224 if ((dim
[number
] & bitvector
) != (olddim
& bitvector
))
226 // The bit is now set and it wasn't before (set)
227 if((dim
[number
] & bitvector
) && (dir
[number
] & bitvector
))
229 tsunami
->intrctrl
->post(number
, TheISA::INTLEVEL_IRQ1
, x
);
230 DPRINTF(Tsunami
, "dim write resulting in posting dir"
231 " interrupt to cpu %d\n", number
);
233 else if ((olddir
& bitvector
) &&
234 !(dir
[number
] & bitvector
))
236 // The bit was set and now its now clear and
237 // we were interrupting on that bit before
238 tsunami
->intrctrl
->clear(number
, TheISA::INTLEVEL_IRQ1
, x
);
239 DPRINTF(Tsunami
, "dim write resulting in clear"
240 " dir interrupt to cpu %d\n", number
);
250 panic("TSDEV_CC_CSR write\n");
252 panic("TSDEV_CC_MTR write not implemented\n");
255 ipreq
= (val
>> 12) & 0xF;
256 //If it is bit 12-15, this is an IPI post
259 supportedWrite
= true;
262 //If it is bit 8-11, this is an IPI clear
264 ipintr
= (val
>> 8) & 0xF;
267 supportedWrite
= true;
270 //If it is the 4-7th bit, clear the RTC interrupt
272 itintr
= (val
>> 4) & 0xF;
275 supportedWrite
= true;
279 if (val
& 0x10000000)
280 supportedWrite
= true;
283 panic("TSDEV_CC_MISC write not implemented\n");
290 panic("TSDEV_CC_AARx write not implemeted\n");
296 if(regnum
== TSDEV_CC_DIM0
)
298 else if(regnum
== TSDEV_CC_DIM1
)
300 else if(regnum
== TSDEV_CC_DIM2
)
309 olddim
= dim
[number
];
310 olddir
= dir
[number
];
312 dir
[number
] = dim
[number
] & drir
;
313 for(int x
= 0; x
< 64; x
++)
315 bitvector
= ULL(1) << x
;
316 // Figure out which bits have changed
317 if ((dim
[number
] & bitvector
) != (olddim
& bitvector
))
319 // The bit is now set and it wasn't before (set)
320 if((dim
[number
] & bitvector
) && (dir
[number
] & bitvector
))
322 tsunami
->intrctrl
->post(number
, TheISA::INTLEVEL_IRQ1
, x
);
323 DPRINTF(Tsunami
, "posting dir interrupt to cpu 0\n");
325 else if ((olddir
& bitvector
) &&
326 !(dir
[number
] & bitvector
))
328 // The bit was set and now its now clear and
329 // we were interrupting on that bit before
330 tsunami
->intrctrl
->clear(number
, TheISA::INTLEVEL_IRQ1
, x
);
331 DPRINTF(Tsunami
, "dim write resulting in clear"
332 " dir interrupt to cpu %d\n",
345 panic("TSDEV_CC_DIR write not implemented\n");
347 panic("TSDEV_CC_DRIR write not implemented\n");
349 panic("TSDEV_CC_PRBEN write not implemented\n");
354 panic("TSDEV_CC_IICx write not implemented\n");
359 panic("TSDEV_CC_MPRx write not implemented\n");
370 panic("default in cchip read reached, accessing 0x%x\n");
372 } // not BIG_TSUNAMI write
373 pkt
.result
= Success
;
378 TsunamiCChip::clearIPI(uint64_t ipintr
)
380 int numcpus
= tsunami
->intrctrl
->cpu
->system
->execContexts
.size();
381 assert(numcpus
<= Tsunami::Max_CPUs
);
384 for (int cpunum
=0; cpunum
< numcpus
; cpunum
++) {
385 // Check each cpu bit
386 uint64_t cpumask
= ULL(1) << cpunum
;
387 if (ipintr
& cpumask
) {
388 // Check if there is a pending ipi
389 if (ipint
& cpumask
) {
391 tsunami
->intrctrl
->clear(cpunum
, TheISA::INTLEVEL_IRQ3
, 0);
392 DPRINTF(IPI
, "clear IPI IPI cpu=%d\n", cpunum
);
395 warn("clear IPI for CPU=%d, but NO IPI\n", cpunum
);
400 panic("Big IPI Clear, but not processors indicated\n");
404 TsunamiCChip::clearITI(uint64_t itintr
)
406 int numcpus
= tsunami
->intrctrl
->cpu
->system
->execContexts
.size();
407 assert(numcpus
<= Tsunami::Max_CPUs
);
410 for (int i
=0; i
< numcpus
; i
++) {
411 uint64_t cpumask
= ULL(1) << i
;
412 if (itintr
& cpumask
& itint
) {
413 tsunami
->intrctrl
->clear(i
, TheISA::INTLEVEL_IRQ2
, 0);
415 DPRINTF(Tsunami
, "clearing rtc interrupt to cpu=%d\n", i
);
420 panic("Big ITI Clear, but not processors indicated\n");
424 TsunamiCChip::reqIPI(uint64_t ipreq
)
426 int numcpus
= tsunami
->intrctrl
->cpu
->system
->execContexts
.size();
427 assert(numcpus
<= Tsunami::Max_CPUs
);
430 for (int cpunum
=0; cpunum
< numcpus
; cpunum
++) {
431 // Check each cpu bit
432 uint64_t cpumask
= ULL(1) << cpunum
;
433 if (ipreq
& cpumask
) {
434 // Check if there is already an ipi (bits 8:11)
435 if (!(ipint
& cpumask
)) {
437 tsunami
->intrctrl
->post(cpunum
, TheISA::INTLEVEL_IRQ3
, 0);
438 DPRINTF(IPI
, "send IPI cpu=%d\n", cpunum
);
441 warn("post IPI for CPU=%d, but IPI already\n", cpunum
);
446 panic("Big IPI Request, but not processors indicated\n");
451 TsunamiCChip::postRTC()
453 int size
= tsunami
->intrctrl
->cpu
->system
->execContexts
.size();
454 assert(size
<= Tsunami::Max_CPUs
);
456 for (int i
= 0; i
< size
; i
++) {
457 uint64_t cpumask
= ULL(1) << i
;
458 if (!(cpumask
& itint
)) {
460 tsunami
->intrctrl
->post(i
, TheISA::INTLEVEL_IRQ2
, 0);
461 DPRINTF(Tsunami
, "Posting RTC interrupt to cpu=%d", i
);
468 TsunamiCChip::postDRIR(uint32_t interrupt
)
470 uint64_t bitvector
= ULL(1) << interrupt
;
471 uint64_t size
= tsunami
->intrctrl
->cpu
->system
->execContexts
.size();
472 assert(size
<= Tsunami::Max_CPUs
);
475 for(int i
=0; i
< size
; i
++) {
476 dir
[i
] = dim
[i
] & drir
;
477 if (dim
[i
] & bitvector
) {
478 tsunami
->intrctrl
->post(i
, TheISA::INTLEVEL_IRQ1
, interrupt
);
479 DPRINTF(Tsunami
, "posting dir interrupt to cpu %d,"
480 "interrupt %d\n",i
, interrupt
);
486 TsunamiCChip::clearDRIR(uint32_t interrupt
)
488 uint64_t bitvector
= ULL(1) << interrupt
;
489 uint64_t size
= tsunami
->intrctrl
->cpu
->system
->execContexts
.size();
490 assert(size
<= Tsunami::Max_CPUs
);
492 if (drir
& bitvector
)
495 for(int i
=0; i
< size
; i
++) {
496 if (dir
[i
] & bitvector
) {
497 tsunami
->intrctrl
->clear(i
, TheISA::INTLEVEL_IRQ1
, interrupt
);
498 DPRINTF(Tsunami
, "clearing dir interrupt to cpu %d,"
499 "interrupt %d\n",i
, interrupt
);
502 dir
[i
] = dim
[i
] & drir
;
506 DPRINTF(Tsunami
, "Spurrious clear? interrupt %d\n", interrupt
);
511 TsunamiCChip::serialize(std::ostream
&os
)
513 SERIALIZE_ARRAY(dim
, Tsunami::Max_CPUs
);
514 SERIALIZE_ARRAY(dir
, Tsunami::Max_CPUs
);
515 SERIALIZE_SCALAR(ipint
);
516 SERIALIZE_SCALAR(itint
);
517 SERIALIZE_SCALAR(drir
);
521 TsunamiCChip::unserialize(Checkpoint
*cp
, const std::string
§ion
)
523 UNSERIALIZE_ARRAY(dim
, Tsunami::Max_CPUs
);
524 UNSERIALIZE_ARRAY(dir
, Tsunami::Max_CPUs
);
525 UNSERIALIZE_SCALAR(ipint
);
526 UNSERIALIZE_SCALAR(itint
);
527 UNSERIALIZE_SCALAR(drir
);
530 BEGIN_DECLARE_SIM_OBJECT_PARAMS(TsunamiCChip
)
532 Param
<Addr
> pio_addr
;
533 Param
<Tick
> pio_latency
;
534 SimObjectParam
<Platform
*> platform
;
535 SimObjectParam
<System
*> system
;
536 SimObjectParam
<Tsunami
*> tsunami
;
538 END_DECLARE_SIM_OBJECT_PARAMS(TsunamiCChip
)
540 BEGIN_INIT_SIM_OBJECT_PARAMS(TsunamiCChip
)
542 INIT_PARAM(pio_addr
, "Device Address"),
543 INIT_PARAM(pio_latency
, "Programmed IO latency"),
544 INIT_PARAM(platform
, "platform"),
545 INIT_PARAM(system
, "system object"),
546 INIT_PARAM(tsunami
, "Tsunami")
548 END_INIT_SIM_OBJECT_PARAMS(TsunamiCChip
)
550 CREATE_SIM_OBJECT(TsunamiCChip
)
552 TsunamiCChip::Params
*p
= new TsunamiCChip::Params
;
553 p
->name
= getInstanceName();
554 p
->pio_addr
= pio_addr
;
555 p
->pio_delay
= pio_latency
;
556 p
->platform
= platform
;
558 p
->tsunami
= tsunami
;
559 return new TsunamiCChip(p
);
562 REGISTER_SIM_OBJECT("TsunamiCChip", TsunamiCChip
)