--- /dev/null
+;
+; int snprintf (char* buf, size_t size, const char* Format, ...);
+;
+; Ullrich von Bassewitz, 2009-09-26
+;
+
+       .export         _snprintf
+       .import         pushax, addysp, decsp6, _vsnprintf
+       .importzp       sp, ptr1
+
+       .macpack        generic
+
+; ----------------------------------------------------------------------------
+; Data
+
+.bss
+
+ParamSize:     .res    1               ; Number of parameter bytes
+
+; ----------------------------------------------------------------------------
+; Code
+
+.code
+
+
+_snprintf:
+       sty     ParamSize               ; Number of param bytes passed in Y
+
+; We have to push buf/size/format, both in the order they already have on stack.
+; To make this somewhat more efficient, we will create space on the stack and
+; then do a copy of the complete block instead of pushing each parameter
+; separately. Since the size of the arguments passed is the same as the size
+; of the fixed arguments, this will allow us to calculate the pointer to the
+; fixed size arguments easier (they're just ParamSize bytes away).
+
+       jsr     decsp6
+
+; Calculate a pointer to the Format argument
+
+       lda     ParamSize
+       add     sp
+       sta     ptr1
+       ldx     sp+1
+       bcc     @L1
+       inx
+@L1:   stx     ptr1+1
+
+; Now copy buf/size/format
+
+       ldy     #6-1
+@L2:   lda     (ptr1),y
+       sta     (sp),y
+       dey
+       bpl     @L2
+
+; Load va_list (last and __fastcall__ parameter to vsprintf)
+
+       lda     ptr1
+       ldx     ptr1+1
+
+; Call vsnprintf
+
+       jsr     _vsnprintf
+
+; Cleanup the stack. We will return what we got from vsprintf
+
+       ldy     ParamSize
+       jmp     addysp
+