/** * 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 #include "mongo/base/parse_number.h" #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/represent_as.h" #include "mongo/util/text.h" namespace mongo { namespace mozjs { const JSFunctionSpec NumberLongInfo::methods[6] = { MONGO_ATTACH_JS_CONSTRAINED_METHOD(toNumber, NumberLongInfo), MONGO_ATTACH_JS_CONSTRAINED_METHOD(toString, NumberLongInfo), MONGO_ATTACH_JS_CONSTRAINED_METHOD(toJSON, 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) getScope(fop)->trackedDelete(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::toJSON::call(JSContext* cx, JS::CallArgs args) { int64_t val = NumberLongInfo::ToNumberLong(cx, args.thisv()); ValueReader(cx, args.rval()) .fromBSON(BSON("$numberLong" << std::to_string(val)), nullptr, false); } 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) { auto numULong = static_cast(NumberLongInfo::ToNumberLong(cx, args.thisv())); ValueReader(cx, args.rval()).fromDouble(numULong >> 32); } void NumberLongInfo::Functions::bottom::call(JSContext* cx, JS::CallArgs args) { auto numULong = static_cast(NumberLongInfo::ToNumberLong(cx, args.thisv())); ValueReader(cx, args.rval()).fromDouble(numULong & 0x00000000FFFFFFFF); } 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()) { auto opt = representAs(arg.toDouble()); uassert(ErrorCodes::BadValue, "number passed to NumberLong must be representable as an int64_t", opt); numLong = *opt; } else if (arg.isString()) { // 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). std::string str = ValueWriter(cx, arg).toString(); // Call parseNumberFromStringWithBase() function to convert string to a number Status status = parseNumberFromStringWithBase(str, 10, &numLong); uassert(ErrorCodes::BadValue, "could not convert string to long long", status.isOK()); } else { numLong = ValueWriter(cx, arg).toInt64(); } } else { uassert(ErrorCodes::BadValue, "floatApprox must be a number", args.get(0).isNumber()); uassert(ErrorCodes::BadValue, "top must be a number", args.get(1).isNumber()); uassert(ErrorCodes::BadValue, "bottom must be a number", args.get(2).isNumber()); auto topOpt = representAs(args.get(1).toNumber()); uassert(ErrorCodes::BadValue, "top must be a 32 bit unsigned number", topOpt); uint64_t top = *topOpt; auto botOpt = representAs(args.get(2).toNumber()); uassert(ErrorCodes::BadValue, "bottom must be a 32 bit unsigned number", botOpt); uint64_t bot = *botOpt; numLong = (top << 32) + bot; } JS_SetPrivate(thisv, scope->trackedNew(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