|
|||||||
|
|
|
![]() |
|
|
Strumenti |
|
|
#1 |
|
Senior Member
Iscritto dal: Nov 2002
Messaggi: 6570
|
[C] Sforamento array
Ciao ragazzi, ho un pezzo di codice che mi dà il seguente errore:
Codice:
Bounds error: attempt to reference memory overrunning the end of an object. Codice:
static opal_atomic_lock_t class_lock = { { OPAL_ATOMIC_UNLOCKED } };
Codice:
static inline int opal_atomic_cmpset_32( volatile int32_t *addr,
int32_t oldval, int32_t newval)
{
unsigned char ret;
__asm__ __volatile (
SMPLOCK "cmpxchgl %1,%2 \n\t"
"sete %0 \n\t"
: "=qm" (ret)
: "q"(newval), "m"(*((volatile long*)addr)), "a"(oldval)
: "memory");
return (int)ret;
}
|
|
|
|
|
|
#2 |
|
Senior Member
Iscritto dal: Nov 2002
Messaggi: 6570
|
Mi sono un pò documentato.
Questa è "Extended Assembler". Queste: Codice:
SMPLOCK "cmpxchgl %1,%2 \n\t"
"sete %0 \n\t"
Questi i registri di output: Codice:
: "=qm" (ret) Codice:
: "q"(newval), "m"(*((volatile long*)addr)), "a"(oldval) Ora, quello che non capisco, i vari %1, %0 si riferiscono ai registri o agli argomenti passati alla funzione? |
|
|
|
|
|
#3 |
|
Senior Member
Iscritto dal: Jun 2002
Città: Dublin
Messaggi: 5989
|
La clobber list è la lista dei registri non passati come parametri di input/output al blocco assembly, ma che sono modificati dalle istruzioni che vi scrivi. In sostanza, dici al compilatore di non fare più affidamento sul contenuto di quei registri. Se usi "memory" fai qualcosa di simile, ma indicandogli che il blocco di codice assembly potrebbe modificare dei valori contenuti nella memoria, oltre ai registri direttamente.
Invece, %0, %1 e %2 fanno riferimento ai parametri di input/output di quel blocco, nel tuo caso sono rispettivamente ret, newval e addr, mentre oldval dovrebbe finire in EAX, se non ricordo male.
__________________
C'ho certi cazzi Mafa' che manco tu che sei pratica li hai visti mai! |
|
|
|
|
|
#4 | |
|
Senior Member
Iscritto dal: Nov 2002
Messaggi: 6570
|
Quote:
Ultima modifica di Unrue : 13-06-2008 alle 09:31. |
|
|
|
|
|
|
#5 |
|
Senior Member
Iscritto dal: Jun 2002
Città: Dublin
Messaggi: 5989
|
SETE sta per Set If Equal, imposta il registro ad 1 se nelle flags è stato impostato il bit Zero dopo l'ultima operazione.
Il valore oldval è usato, ma in maniera un po' più nascosta: viene caricato in EAX (è questo il significato di "a") e poi usato da CMPXCHGL. Questa infatti confronta il valore in EAX col secondo operando: se sono uguali, imposta Zero e carica il primo operando nel secondo, altrimenti disattiva Zero e carica il secondo operando in AL (nota che se la sintassi fosse quella Intel anziché AT&T, dovresti invertire sempre primo/secondo operando). In sostanza, si sta facendo una cosa tipo: Codice:
long *pMemory = (long *)addr;
if ((*pMemory) == oldval)
{
(*pMemory) = newval;
return 1;
}
return 0;
__________________
C'ho certi cazzi Mafa' che manco tu che sei pratica li hai visti mai! |
|
|
|
|
|
#6 | |
|
Senior Member
Iscritto dal: Nov 2002
Messaggi: 6570
|
Quote:
Altra cosa, io sto lavorando su un amd 64 bit, ma come mai quindi in questo codice c'è la sintassi AT&T e non Intel? Inoltre, siccome quella parte di codice lavora su valori a 32 bit, che senso potrebbe avere quel cast a long? Trallaltro è proprio nella riga incriminata. Ultima modifica di Unrue : 13-06-2008 alle 10:45. |
|
|
|
|
|
|
#7 |
|
Senior Member
Iscritto dal: Jun 2002
Città: Dublin
Messaggi: 5989
|
Sì, la CMPXCHGL imposta una flag, oltre a spostare i valori. Poi con SETE quel codice aggiorna il valore dentro una variabile locale.
Perché la sintassi AT&T? Non so, probabilmente solo una questione di compilatore, con GCC al momento è l'unica supportata, da quel che mi risulta. ciao
__________________
C'ho certi cazzi Mafa' che manco tu che sei pratica li hai visti mai! |
|
|
|
|
| Strumenti | |
|
|
Tutti gli orari sono GMT +1. Ora sono le: 10:38.



















