/** * Copyright (C) 2015 MongoDB Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License, version 3, * as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . * * As a special exception, the copyright holders give permission to link the * code of portions of this program with the OpenSSL library under certain * conditions as described in each individual source file and distribute * linked combinations including the program with the OpenSSL library. You * must comply with the GNU Affero General Public License in all respects * for all of the code used other than as permitted herein. If you modify * file(s) with this exception, you may extend this exception to your * version of the file(s), but you are not obligated to do so. If you do not * wish to do so, delete this exception statement from your version. If you * delete this exception statement from all source files in the program, * then also delete it in the license file. */ #include "mongo/platform/basic.h" #include "mongo/scripting/mozjs/numberlong.h" #include #include "mongo/scripting/mozjs/implscope.h" #include "mongo/scripting/mozjs/objectwrapper.h" #include "mongo/scripting/mozjs/valuereader.h" #include "mongo/scripting/mozjs/valuewriter.h" #include "mongo/scripting/mozjs/wrapconstrainedmethod.h" #include "mongo/util/mongoutils/str.h" #include "mongo/util/text.h" namespace mongo { namespace mozjs { const JSFunctionSpec NumberLongInfo::methods[5] = { MONGO_ATTACH_JS_CONSTRAINED_METHOD(toNumber, NumberLongInfo), MONGO_ATTACH_JS_CONSTRAINED_METHOD(toString, NumberLongInfo), MONGO_ATTACH_JS_CONSTRAINED_METHOD(valueOf, NumberLongInfo), MONGO_ATTACH_JS_CONSTRAINED_METHOD(compare, NumberLongInfo), JS_FS_END}; const char* const NumberLongInfo::className = "NumberLong"; void NumberLongInfo::finalize(JSFreeOp* fop, JSObject* obj) { auto numLong = static_cast(JS_GetPrivate(obj)); if (numLong) delete numLong; } int64_t NumberLongInfo::ToNumberLong(JSContext* cx, JS::HandleValue thisv) { auto numLong = static_cast(JS_GetPrivate(thisv.toObjectOrNull())); return numLong ? *numLong : 0; } int64_t NumberLongInfo::ToNumberLong(JSContext* cx, JS::HandleObject thisv) { auto numLong = static_cast(JS_GetPrivate(thisv)); return numLong ? *numLong : 0; } void NumberLongInfo::Functions::valueOf::call(JSContext* cx, JS::CallArgs args) { int64_t out = NumberLongInfo::ToNumberLong(cx, args.thisv()); ValueReader(cx, args.rval()).fromDouble(out); } void NumberLongInfo::Functions::toNumber::call(JSContext* cx, JS::CallArgs args) { valueOf::call(cx, args); } void NumberLongInfo::Functions::toString::call(JSContext* cx, JS::CallArgs args) { str::stream ss; int64_t val = NumberLongInfo::ToNumberLong(cx, args.thisv()); const int64_t limit = 2LL << 30; if (val <= -limit || limit <= val) ss << "NumberLong(\"" << val << "\")"; else ss << "NumberLong(" << val << ")"; ValueReader(cx, args.rval()).fromStringData(ss.operator std::string()); } void NumberLongInfo::Functions::compare::call(JSContext* cx, JS::CallArgs args) { uassert(ErrorCodes::BadValue, "NumberLong.compare() needs 1 argument", args.length() == 1); uassert(ErrorCodes::BadValue, "NumberLong.compare() argument must be a NumberLong", getScope(cx)->getProto().instanceOf(args.get(0))); int64_t thisVal = NumberLongInfo::ToNumberLong(cx, args.thisv()); int64_t otherVal = NumberLongInfo::ToNumberLong(cx, args.get(0)); int comparison = 0; if (thisVal < otherVal) { comparison = -1; } else if (thisVal > otherVal) { comparison = 1; } ValueReader(cx, args.rval()).fromDouble(comparison); } void NumberLongInfo::Functions::floatApprox::call(JSContext* cx, JS::CallArgs args) { int64_t numLong = NumberLongInfo::ToNumberLong(cx, args.thisv()); ValueReader(cx, args.rval()).fromDouble(numLong); } void NumberLongInfo::Functions::top::call(JSContext* cx, JS::CallArgs args) { int64_t numLong = NumberLongInfo::ToNumberLong(cx, args.thisv()); // values above 2^53 are not accurately represented in JS if (numLong == INT64_MIN || std::abs(numLong) >= 9007199254740992LL) { auto val64 = static_cast(numLong); ValueReader(cx, args.rval()).fromDouble(val64 >> 32); } else { args.rval().setUndefined(); } } void NumberLongInfo::Functions::bottom::call(JSContext* cx, JS::CallArgs args) { int64_t numLong = NumberLongInfo::ToNumberLong(cx, args.thisv()); // values above 2^53 are not accurately represented in JS if (numLong == INT64_MIN || std::abs(numLong) >= 9007199254740992LL) { auto val64 = static_cast(numLong); ValueReader(cx, args.rval()).fromDouble(val64 & 0x00000000FFFFFFFF); } else { args.rval().setUndefined(); } } void NumberLongInfo::construct(JSContext* cx, JS::CallArgs args) { uassert(ErrorCodes::BadValue, "NumberLong needs 0, 1 or 3 arguments", args.length() == 0 || args.length() == 1 || args.length() == 3); auto scope = getScope(cx); JS::RootedObject thisv(cx); scope->getProto().newObject(&thisv); int64_t numLong; ObjectWrapper o(cx, thisv); if (args.length() == 0) { numLong = 0; } else if (args.length() == 1) { auto arg = args.get(0); if (arg.isInt32()) { numLong = arg.toInt32(); } else if (arg.isDouble()) { numLong = arg.toDouble(); } else { std::string str = ValueWriter(cx, arg).toString(); int64_t val; // For string values we call strtoll because we expect non-number string // values to fail rather than return 0 (which is the behavior of ToInt64). if (arg.isString()) val = parseLL(str.c_str()); // Otherwise we call the toNumber on the js value to get the int64_t value. else val = ValueWriter(cx, arg).toInt64(); numLong = val; } } else { if (!args.get(0).isNumber()) uasserted(ErrorCodes::BadValue, "floatApprox must be a number"); double top = 0; if (args.get(1).isNumber()) top = static_cast(static_cast(args.get(1).toNumber())); if (!args.get(1).isNumber() || args.get(1).toNumber() != top) uasserted(ErrorCodes::BadValue, "top must be a 32 bit unsigned number"); double bot = 0; if (args.get(2).isNumber()) bot = static_cast(static_cast(args.get(2).toNumber())); if (!args.get(2).isNumber() || args.get(2).toNumber() != bot) uasserted(ErrorCodes::BadValue, "bottom must be a 32 bit unsigned number"); numLong = (static_cast(top) << 32) + static_cast(bot); } JS_SetPrivate(thisv, new int64_t(numLong)); args.rval().setObjectOrNull(thisv); } void NumberLongInfo::postInstall(JSContext* cx, JS::HandleObject global, JS::HandleObject proto) { JS::RootedValue undef(cx); undef.setUndefined(); // floatapprox if (!JS_DefinePropertyById( cx, proto, getScope(cx)->getInternedStringId(InternedString::floatApprox), undef, JSPROP_ENUMERATE | JSPROP_SHARED, smUtils::wrapConstrainedMethod, nullptr)) { uasserted(ErrorCodes::JSInterpreterFailure, "Failed to JS_DefinePropertyById"); } // top if (!JS_DefinePropertyById(cx, proto, getScope(cx)->getInternedStringId(InternedString::top), undef, JSPROP_ENUMERATE | JSPROP_SHARED, smUtils::wrapConstrainedMethod, nullptr)) { uasserted(ErrorCodes::JSInterpreterFailure, "Failed to JS_DefinePropertyById"); } // bottom if (!JS_DefinePropertyById( cx, proto, getScope(cx)->getInternedStringId(InternedString::bottom), undef, JSPROP_ENUMERATE | JSPROP_SHARED, smUtils::wrapConstrainedMethod, nullptr)) { uasserted(ErrorCodes::JSInterpreterFailure, "Failed to JS_DefinePropertyById"); } } } // namespace mozjs } // namespace mongo