ref: 442b8d27a1ef17c3127a4ca58f2df62523243270
dir: /jsintern.c/
#include "js.h" #include "jsstate.h" /* Use an AA-tree to quickly look up interned strings. */ struct js_StringNode { char *string; js_StringNode *left, *right; int level; }; static js_StringNode sentinel = { "", &sentinel, &sentinel, 0 }; static js_StringNode *newstringnode(const char *string, const char **result) { js_StringNode *node = malloc(sizeof(js_StringNode)); node->string = strdup(string); node->left = node->right = &sentinel; node->level = 1; return *result = node->string, node; } static js_StringNode *skew(js_StringNode *node) { if (node->level != 0) { if (node->left->level == node->level) { js_StringNode *save = node; node = node->left; save->left = node->right; node->right = save; } node->right = skew(node->right); } return node; } static js_StringNode *split(js_StringNode *node) { if (node->level != 0 && node->right->right->level == node->level) { js_StringNode *save = node; node = node->right; save->right = node->left; node->left = save; node->level++; node->right = split(node->right); } return node; } static js_StringNode *insert(js_StringNode *node, const char *string, const char **result) { if (node != &sentinel) { int c = strcmp(string, node->string); if (c < 0) node->left = insert(node->left, string, result); else if (c > 0) node->right = insert(node->right, string, result); else return *result = node->string, node; node = skew(node); node = split(node); return node; } return newstringnode(string, result); } static void printstringnode(js_StringNode *node, int level) { int i; if (node->left != &sentinel) printstringnode(node->left, level + 1); printf("%d: ", node->level); for (i = 0; i < level; ++i) putchar('\t'); printf("'%s'\n", node->string); if (node->right != &sentinel) printstringnode(node->right, level + 1); } void js_printstrings(js_State *J) { js_StringNode *root = J->strings; printf("--- string dump ---\n"); if (root && root != &sentinel) printstringnode(root, 0); printf("---\n"); } static void js_freestringnode(js_State *J, js_StringNode *node) { if (node->left != &sentinel) js_freestringnode(J, node->left); if (node->right != &sentinel) js_freestringnode(J, node->right); free(node->string); free(node); } void js_freestrings(js_State *J) { if (J->strings && J->strings != &sentinel) js_freestringnode(J, J->strings); } const char *js_intern(js_State *J, const char *s) { const char *result; if (!J->strings) J->strings = &sentinel; J->strings = insert(J->strings, s, &result); return result; }