mirror of
https://github.com/jart/cosmopolitan.git
synced 2025-01-31 11:37:35 +00:00
5660ec4741
This release is an atomic upgrade to GCC 14.1.0 with C23 and C++23
127 lines
5.1 KiB
C
127 lines
5.1 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│ vi: set et ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi │
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2020 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ Permission to use, copy, modify, and/or distribute this software for │
|
|
│ any purpose with or without fee is hereby granted, provided that the │
|
|
│ above copyright notice and this permission notice appear in all copies. │
|
|
│ │
|
|
│ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │
|
|
│ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │
|
|
│ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │
|
|
│ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │
|
|
│ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │
|
|
│ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │
|
|
│ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │
|
|
│ PERFORMANCE OF THIS SOFTWARE. │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "libc/fmt/itoa.h"
|
|
#include "libc/intrin/safemacros.h"
|
|
#include "libc/mem/gc.h"
|
|
#include "libc/str/str.h"
|
|
#include "libc/str/strwidth.h"
|
|
#include "libc/x/x.h"
|
|
#include "tool/viz/lib/formatstringtable.h"
|
|
|
|
#define STREQ(A, B) (strcasecmp(A, B) == 0)
|
|
|
|
static int GetArrayAlignment(long yn, long xn, int w, int align) {
|
|
/* abi guaranteed to 16 after which gcc / clang disagree */
|
|
int i, r;
|
|
r = i = align;
|
|
while ((i *= 2) <= __BIGGEST_ALIGNMENT__) {
|
|
if (yn * xn * w >= i) {
|
|
r = i;
|
|
}
|
|
}
|
|
return r;
|
|
}
|
|
|
|
static const char *GetStorageSpecifier(const char *type, int *out_width,
|
|
int *out_align) {
|
|
if (STREQ(type, "unsigned char") || STREQ(type, "char") ||
|
|
STREQ(type, "signed char")) {
|
|
*out_width = 1;
|
|
*out_align = 1;
|
|
return "byte";
|
|
} else if (STREQ(type, "unsigned short") || STREQ(type, "short") ||
|
|
STREQ(type, "signed short")) {
|
|
*out_width = 2;
|
|
*out_align = 2;
|
|
return "short";
|
|
} else if (STREQ(type, "unsigned int") || STREQ(type, "unsigned") ||
|
|
STREQ(type, "int") || STREQ(type, "signed int")) {
|
|
*out_width = 4;
|
|
*out_align = 4;
|
|
return "long";
|
|
} else if (STREQ(type, "unsigned long") || STREQ(type, "unsigned") ||
|
|
STREQ(type, "long") || STREQ(type, "signed long")) {
|
|
*out_width = 8;
|
|
*out_align = 8;
|
|
return "quad";
|
|
} else if (STREQ(type, "float")) {
|
|
*out_width = 4;
|
|
*out_align = 4;
|
|
return "float";
|
|
} else if (STREQ(type, "double")) {
|
|
*out_width = 8;
|
|
*out_align = 8;
|
|
return "double";
|
|
} else {
|
|
*out_width = __BIGGEST_ALIGNMENT__;
|
|
*out_align = __BIGGEST_ALIGNMENT__;
|
|
return type;
|
|
}
|
|
}
|
|
|
|
static void EmitSection(long yn, long xn, int w, int arrayalign, int emit(),
|
|
void *a) {
|
|
void (*Emit)(const char *, void *) = (void (*)(const char *, void *))emit;
|
|
char alignstr[21];
|
|
FormatUint32(alignstr, arrayalign);
|
|
if (arrayalign <= 8 && yn * xn * w == 8) {
|
|
Emit("\t.rodata.cst", a);
|
|
Emit("8\n", a);
|
|
} else if (arrayalign <= 16 && yn * xn * w == 16) {
|
|
Emit("\t.rodata.cst", a);
|
|
Emit("16\n", a);
|
|
} else if (arrayalign <= 32 && yn * xn * w == 32) {
|
|
Emit("\t.rodata.cst", a);
|
|
Emit("32\n", a);
|
|
} else if (arrayalign <= 64 && yn * xn * w == 64) {
|
|
Emit("\t.rodata.cst", a);
|
|
Emit("64\n", a);
|
|
} else {
|
|
Emit("\t.rodata\n", a);
|
|
Emit("\t.align\t", a);
|
|
Emit(alignstr, a);
|
|
Emit("\n", a);
|
|
}
|
|
}
|
|
|
|
void *FormatStringTableAsAssembly(long yn, long xn, const char *const T[yn][xn],
|
|
int emit(), void *a, const char *type,
|
|
const char *name, const char *scope) {
|
|
void (*Emit)(const char *, void *) = (void (*)(const char *, void *))emit;
|
|
int w, align;
|
|
const char *storage;
|
|
char ynstr[21], xnstr[21];
|
|
name = firstnonnull(name, "M");
|
|
storage = GetStorageSpecifier(firstnonnull(type, "long"), &w, &align);
|
|
FormatUint64(ynstr, yn);
|
|
FormatUint64(xnstr, xn);
|
|
EmitSection(yn, xn, w, GetArrayAlignment(yn, xn, w, align), emit, a);
|
|
Emit(name, a);
|
|
Emit(":", a);
|
|
if (strwidth(name, 0) >= 8)
|
|
Emit("\n", a);
|
|
FormatStringTable(yn, xn, T, emit, a, gc(xstrcat("\t.", storage, "\t")), ",",
|
|
"\n");
|
|
Emit("\t.endobj\t", a);
|
|
Emit(name, a);
|
|
Emit(",", a);
|
|
Emit(firstnonnull(scope, "globl"), a);
|
|
Emit("\n\t.previous\n", a);
|
|
return (/* unconst */ void *)T;
|
|
}
|