/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include <stdio.h>
typedef unsigned nsresult;
typedef unsigned uint32_t;
typedef unsigned nsXPCVariant;
#if defined (WIN32)
# define NS_IMETHOD
virtual nsresult __stdcall
# define NS_IMETHODIMP nsresult __stdcall
#else
# define NS_IMETHOD
virtual nsresult
# define NS_IMETHODIMP nsresult
#endif
class base {
public :
NS_IMETHOD ignored() =
0 ;
};
class foo :
public base {
public :
NS_IMETHOD callme1(
int i,
int j) =
0 ;
NS_IMETHOD callme2(
int i,
int j) =
0 ;
NS_IMETHOD callme3(
int i,
int j) =
0 ;
};
class bar :
public foo {
public :
NS_IMETHOD ignored() override;
NS_IMETHOD callme1(
int i,
int j) override;
NS_IMETHOD callme2(
int i,
int j) override;
NS_IMETHOD callme3(
int i,
int j) override;
};
class baz :
public base {
public :
NS_IMETHOD ignored() override;
NS_IMETHOD callme1() override;
NS_IMETHOD callme2() override;
NS_IMETHOD callme3() override;
void setfoo(foo* f) { other = f; }
foo* other;
};
NS_IMETHODIMP baz::ignored() {
return 0 ; }
NS_IMETHODIMP bar::ignored() {
return 0 ; }
NS_IMETHODIMP bar::callme1(
int i,
int j) {
printf(
"called bar::callme1 with: %d %d\n" , i, j);
return 15 ;
}
NS_IMETHODIMP bar::callme2(
int i,
int j) {
printf(
"called bar::callme2 with: %d %d\n" , i, j);
return 25 ;
}
NS_IMETHODIMP bar::callme3(
int i,
int j) {
printf(
"called bar::callme3 with: %d %d\n" , i, j);
return 35 ;
}
void docall(foo* f,
int i,
int j) { f->callme1(i, j); }
/***************************************************************************/
#if defined (WIN32)
static int __stdcall PrepareAndDispatch(baz* self, uint32_t methodIndex,
uint32_t* args,
uint32_t* stackBytesToPop) {
fprintf(stdout,
"PrepareAndDispatch (%p, %d, %p)\n" , (
void *)self, methodIndex,
(
void *)args);
foo* a = self->other;
int p1 = (
int )*args;
int p2 = (
int )*(args +
1 );
int out =
0 ;
switch (methodIndex) {
case 1 :
out = a->callme1(p1, p2);
break ;
case 2 :
out = a->callme2(p1, p2);
break ;
case 3 :
out = a->callme3(p1, p2);
break ;
}
*stackBytesToPop =
2 *
4 ;
return out;
}
# ifndef __GNUC__
static __declspec(naked)
void SharedStub(
void ) {
__
asm {
push ebp
// set up simple stack frame
mov ebp, esp
// stack has: ebp/vtbl_index/retaddr/this/args
push ecx
// make room for a ptr
lea eax, [ebp-
4 ]
// pointer to stackBytesToPop
push eax
lea ecx, [ebp+
16 ]
// pointer to args
push ecx
mov edx, [ebp+
4 ]
// vtbl_index
push edx
mov eax, [ebp+
12 ]
// this
push eax
call PrepareAndDispatch
mov edx, [ebp+
8 ]
// return address
mov ecx, [ebp-
4 ]
// stackBytesToPop
add ecx,
12 // for this, the index, and ret address
mov esp, ebp
pop ebp
add esp, ecx
// fix up stack pointer
jmp edx
// simulate __stdcall return
}
}
// these macros get expanded (many times) in the file #included below
# define STUB_ENTRY(n) \
__declspec(naked) nsresult __stdcall baz::callme
## n() { \
__
asm push n __
asm jmp SharedStub \
}
# else /* __GNUC__ */
# define STUB_ENTRY(n) \
nsresult __stdcall baz::callme
## n() { \
uint32_t *args, stackBytesToPop; \
int result =
0 ; \
baz* obj; \
__asm__ __volatile__( \
"leal 0x0c(%%ebp), %0\n\t" /* args */ \
"movl 0x08(%%ebp), %1\n\t" /* this */ \
:
"=r" (args),
"=r" (obj)); \
result = PrepareAndDispatch(obj, n, args, &stackBytesToPop); \
fprintf(stdout,
"stub returning: %d\n" , result); \
fprintf(stdout,
"bytes to pop: %d\n" , stackBytesToPop); \
return result; \
}
# endif
/* ! __GNUC__ */
#else
/***************************************************************************/
// just Linux_x86 now. Add other later...
static int PrepareAndDispatch(baz* self, uint32_t methodIndex, uint32_t* args) {
foo* a = self->other;
int p1 = (
int )*args;
int p2 = (
int )*(args +
1 );
switch (methodIndex) {
case 1 :
a->callme1(p1, p2);
break ;
case 2 :
a->callme2(p1, p2);
break ;
case 3 :
a->callme3(p1, p2);
break ;
}
return 1 ;
}
# define STUB_ENTRY(n) \
nsresult baz::callme
## n() { \
void * method = PrepareAndDispatch; \
nsresult result; \
__asm__ __volatile__( \
"leal 0x0c(%%ebp), %%ecx\n\t" /* args */ \
"pushl %%ecx\n\t" \
"pushl $" #n \
"\n\t" /* method index */ \
"movl 0x08(%%ebp), %%ecx\n\t" /* this */ \
"pushl %%ecx\n\t" \
"call *%%edx" /* PrepareAndDispatch */ \
:
"=a" (result)
/* %0 */ \
:
"d" (method)
/* %1 */ \
:
"memory" ); \
return result; \
}
#endif
/***************************************************************************/
STUB_ENTRY(
1 )
STUB_ENTRY(
2 )
STUB_ENTRY(
3 )
int main() {
foo* a =
new bar();
baz* b =
new baz();
/* here we make the global 'check for alloc failure' checker happy */
if (!a || !b)
return 1 ;
foo* c = (foo*)b;
b->setfoo(a);
c->callme1(
1 ,
2 );
c->callme2(
2 ,
4 );
c->callme3(
3 ,
6 );
return 0 ;
}
Messung V0.5 in Prozent C=86 H=96 G=90
¤ Dauer der Verarbeitung: 0.11 Sekunden
(vorverarbeitet am 2026-06-04)
¤
*© Formatika GbR, Deutschland