aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--Changelog.md1
-rw-r--r--libdevcore/FixedHash.h142
-rw-r--r--libevmasm/AssemblyItem.cpp5
-rw-r--r--libsolidity/analysis/TypeChecker.cpp22
-rwxr-xr-xscripts/travis-emscripten/build_emscripten.sh25
-rw-r--r--test/libsolidity/SolidityNameAndTypeResolution.cpp44
6 files changed, 83 insertions, 156 deletions
diff --git a/Changelog.md b/Changelog.md
index c3482c4b..670182af 100644
--- a/Changelog.md
+++ b/Changelog.md
@@ -2,6 +2,7 @@
Features:
* Optimizer: Add new optimization step to remove unused ``JUMPDEST``s.
+ * Type Checker: Warn on using literals as tight packing parameters in ``keccak256``, ``sha3``, ``sha256`` and ``ripemd160``.
Bugfixes:
diff --git a/libdevcore/FixedHash.h b/libdevcore/FixedHash.h
index 5bc70095..cd6e1da1 100644
--- a/libdevcore/FixedHash.h
+++ b/libdevcore/FixedHash.h
@@ -31,16 +31,10 @@
#include <array>
#include <cstdint>
#include <algorithm>
-#include <set>
-#include <unordered_set>
namespace dev
{
-/// Compile-time calculation of Log2 of constant values.
-template <unsigned N> struct StaticLog2 { enum { result = 1 + StaticLog2<N/2>::result }; };
-template <> struct StaticLog2<1> { enum { result = 0 }; };
-
/// Fixed-size raw-byte array container type, with an API optimised for storing hashes.
/// Transparently converts to/from the corresponding arithmetic type; this will
/// assume the data contained in the hash is big-endian.
@@ -54,9 +48,6 @@ public:
/// The size of the container.
enum { size = N };
- /// A dummy flag to avoid accidental construction from pointer.
- enum ConstructFromPointerType { ConstructFromPointer };
-
/// Method to convert from a string.
enum ConstructFromStringType { FromHex, FromBinary };
@@ -81,9 +72,6 @@ public:
/// Explicitly construct, copying from a byte array.
explicit FixedHash(bytesConstRef _b, ConstructFromHashType _t = FailIfDifferent) { if (_b.size() == N) memcpy(m_data.data(), _b.data(), std::min<unsigned>(_b.size(), N)); else { m_data.fill(0); if (_t != FailIfDifferent) { auto c = std::min<unsigned>(_b.size(), N); for (unsigned i = 0; i < c; ++i) m_data[_t == AlignRight ? N - 1 - i : i] = _b[_t == AlignRight ? _b.size() - 1 - i : i]; } } }
- /// Explicitly construct, copying from a bytes in memory with given pointer.
- explicit FixedHash(byte const* _bs, ConstructFromPointerType) { memcpy(m_data.data(), _bs, N); }
-
/// Explicitly construct, copying from a string.
explicit FixedHash(std::string const& _s, ConstructFromStringType _t = FromHex, ConstructFromHashType _ht = FailIfDifferent): FixedHash(_t == FromHex ? fromHex(_s, WhenError::Throw) : dev::asBytes(_s), _ht) {}
@@ -96,37 +84,16 @@ public:
// The obvious comparison operators.
bool operator==(FixedHash const& _c) const { return m_data == _c.m_data; }
bool operator!=(FixedHash const& _c) const { return m_data != _c.m_data; }
+ /// Required to sort objects of this type or use them as map keys.
bool operator<(FixedHash const& _c) const { for (unsigned i = 0; i < N; ++i) if (m_data[i] < _c.m_data[i]) return true; else if (m_data[i] > _c.m_data[i]) return false; return false; }
- bool operator>=(FixedHash const& _c) const { return !operator<(_c); }
- bool operator<=(FixedHash const& _c) const { return operator==(_c) || operator<(_c); }
- bool operator>(FixedHash const& _c) const { return !operator<=(_c); }
-
- // The obvious binary operators.
- FixedHash& operator^=(FixedHash const& _c) { for (unsigned i = 0; i < N; ++i) m_data[i] ^= _c.m_data[i]; return *this; }
- FixedHash operator^(FixedHash const& _c) const { return FixedHash(*this) ^= _c; }
- FixedHash& operator|=(FixedHash const& _c) { for (unsigned i = 0; i < N; ++i) m_data[i] |= _c.m_data[i]; return *this; }
- FixedHash operator|(FixedHash const& _c) const { return FixedHash(*this) |= _c; }
- FixedHash& operator&=(FixedHash const& _c) { for (unsigned i = 0; i < N; ++i) m_data[i] &= _c.m_data[i]; return *this; }
- FixedHash operator&(FixedHash const& _c) const { return FixedHash(*this) &= _c; }
- FixedHash operator~() const { FixedHash ret; for (unsigned i = 0; i < N; ++i) ret[i] = ~m_data[i]; return ret; }
-
- // Big-endian increment.
- FixedHash& operator++() { for (unsigned i = size; i > 0 && !++m_data[--i]; ) {} return *this; }
- /// @returns true if all one-bits in @a _c are set in this object.
- bool contains(FixedHash const& _c) const { return (*this & _c) == _c; }
+ FixedHash operator~() const { FixedHash ret; for (unsigned i = 0; i < N; ++i) ret[i] = ~m_data[i]; return ret; }
/// @returns a particular byte from the hash.
byte& operator[](unsigned _i) { return m_data[_i]; }
/// @returns a particular byte from the hash.
byte operator[](unsigned _i) const { return m_data[_i]; }
- /// @returns an abridged version of the hash as a user-readable hex string.
- std::string abridged() const { return toHex(ref().cropped(0, 4)) + "\342\200\246"; }
-
- /// @returns a version of the hash as a user-readable hex string that leaves out the middle part.
- std::string abridgedMiddle() const { return toHex(ref().cropped(0, 4)) + "\342\200\246" + toHex(ref().cropped(N - 4)); }
-
/// @returns the hash as a user-readable hex string.
std::string hex() const { return toHex(ref()); }
@@ -151,54 +118,17 @@ public:
/// @returns a constant reference to the object's data as an STL array.
std::array<byte, N> const& asArray() const { return m_data; }
- struct hash
- {
- /// Make a hash of the object's data.
- size_t operator()(FixedHash const& _value) const { return boost::hash_range(_value.m_data.cbegin(), _value.m_data.cend()); }
- };
-
- template <unsigned P, unsigned M> inline FixedHash& shiftBloom(FixedHash<M> const& _h)
- {
- return (*this |= _h.template bloomPart<P, N>());
- }
-
- template <unsigned P, unsigned M> inline bool containsBloom(FixedHash<M> const& _h)
- {
- return contains(_h.template bloomPart<P, N>());
- }
-
- template <unsigned P, unsigned M> inline FixedHash<M> bloomPart() const
- {
- unsigned const c_bloomBits = M * 8;
- unsigned const c_mask = c_bloomBits - 1;
- unsigned const c_bloomBytes = (StaticLog2<c_bloomBits>::result + 7) / 8;
-
- static_assert((M & (M - 1)) == 0, "M must be power-of-two");
- static_assert(P * c_bloomBytes <= N, "out of range");
-
- FixedHash<M> ret;
- byte const* p = data();
- for (unsigned i = 0; i < P; ++i)
- {
- unsigned index = 0;
- for (unsigned j = 0; j < c_bloomBytes; ++j, ++p)
- index = (index << 8) | *p;
- index &= c_mask;
- ret[M - 1 - index / 8] |= (1 << (index % 8));
- }
- return ret;
- }
-
/// Returns the index of the first bit set to one, or size() * 8 if no bits are set.
inline unsigned firstBitSet() const
{
unsigned ret = 0;
for (auto d: m_data)
if (d)
+ {
for (;; ++ret, d <<= 1)
if (d & 0x80)
return ret;
- else {}
+ }
else
ret += 8;
return ret;
@@ -210,21 +140,6 @@ private:
std::array<byte, N> m_data; ///< The binary data.
};
-/// Fast equality operator for h256.
-template<> inline bool FixedHash<32>::operator==(FixedHash<32> const& _other) const
-{
- const uint64_t* hash1 = (const uint64_t*)data();
- const uint64_t* hash2 = (const uint64_t*)_other.data();
- return (hash1[0] == hash2[0]) && (hash1[1] == hash2[1]) && (hash1[2] == hash2[2]) && (hash1[3] == hash2[3]);
-}
-
-/// Fast std::hash compatible hash function object for h256.
-template<> inline size_t FixedHash<32>::hash::operator()(FixedHash<32> const& value) const
-{
- uint64_t const* data = reinterpret_cast<uint64_t const*>(value.data());
- return boost::hash_range(data, data + 4);
-}
-
/// Stream I/O for the FixedHash class.
template <unsigned N>
inline std::ostream& operator<<(std::ostream& _out, FixedHash<N> const& _h)
@@ -238,56 +153,7 @@ inline std::ostream& operator<<(std::ostream& _out, FixedHash<N> const& _h)
}
// Common types of FixedHash.
-using h2048 = FixedHash<256>;
-using h1024 = FixedHash<128>;
-using h520 = FixedHash<65>;
-using h512 = FixedHash<64>;
using h256 = FixedHash<32>;
using h160 = FixedHash<20>;
-using h128 = FixedHash<16>;
-using h64 = FixedHash<8>;
-using h512s = std::vector<h512>;
-using h256s = std::vector<h256>;
-using h160s = std::vector<h160>;
-using h256Set = std::set<h256>;
-using h160Set = std::set<h160>;
-using h256Hash = std::unordered_set<h256>;
-using h160Hash = std::unordered_set<h160>;
-
-/// Convert the given value into h160 (160-bit unsigned integer) using the right 20 bytes.
-inline h160 right160(h256 const& _t)
-{
- h160 ret;
- memcpy(ret.data(), _t.data() + 12, 20);
- return ret;
-}
-/// Convert the given value into h160 (160-bit unsigned integer) using the left 20 bytes.
-inline h160 left160(h256 const& _t)
-{
- h160 ret;
- memcpy(&ret[0], _t.data(), 20);
- return ret;
-}
-
-inline std::string toString(h256s const& _bs)
-{
- std::ostringstream out;
- out << "[ ";
- for (auto i: _bs)
- out << i.abridged() << ", ";
- out << "]";
- return out.str();
-}
-
-}
-
-namespace std
-{
- /// Forward std::hash<dev::FixedHash> to dev::FixedHash::hash.
- template<> struct hash<dev::h64>: dev::h64::hash {};
- template<> struct hash<dev::h128>: dev::h128::hash {};
- template<> struct hash<dev::h160>: dev::h160::hash {};
- template<> struct hash<dev::h256>: dev::h256::hash {};
- template<> struct hash<dev::h512>: dev::h512::hash {};
}
diff --git a/libevmasm/AssemblyItem.cpp b/libevmasm/AssemblyItem.cpp
index 419a8c0b..1af266b6 100644
--- a/libevmasm/AssemblyItem.cpp
+++ b/libevmasm/AssemblyItem.cpp
@@ -249,8 +249,11 @@ ostream& dev::eth::operator<<(ostream& _out, AssemblyItem const& _item)
_out << " PushProgramSize";
break;
case PushLibraryAddress:
- _out << " PushLibraryAddress " << hex << h256(_item.data()).abridgedMiddle() << dec;
+ {
+ string hash(h256((_item.data())).hex());
+ _out << " PushLibraryAddress " << hash.substr(0, 8) + "..." + hash.substr(hash.length() - 8);
break;
+ }
case UndefinedItem:
_out << " ???";
break;
diff --git a/libsolidity/analysis/TypeChecker.cpp b/libsolidity/analysis/TypeChecker.cpp
index 99f3c64c..d594a060 100644
--- a/libsolidity/analysis/TypeChecker.cpp
+++ b/libsolidity/analysis/TypeChecker.cpp
@@ -1446,6 +1446,28 @@ bool TypeChecker::visit(FunctionCall const& _functionCall)
_functionCall.annotation().type = make_shared<TupleType>(functionType->returnParameterTypes());
TypePointers parameterTypes = functionType->parameterTypes();
+
+ if (!functionType->padArguments())
+ {
+ for (size_t i = 0; i < arguments.size(); ++i)
+ {
+ auto const& argType = type(*arguments[i]);
+ if (auto literal = dynamic_cast<RationalNumberType const*>(argType.get()))
+ {
+ /* If no mobile type is available an error will be raised elsewhere. */
+ if (literal->mobileType())
+ m_errorReporter.warning(
+ _functionCall.location(),
+ "The type of \"" +
+ argType->toString() +
+ "\" was inferred as " +
+ literal->mobileType()->toString() +
+ ". This is probably not desired. Use an explicit type to silence this warning."
+ );
+ }
+ }
+ }
+
if (!functionType->takesArbitraryParameters() && parameterTypes.size() != arguments.size())
{
string msg =
diff --git a/scripts/travis-emscripten/build_emscripten.sh b/scripts/travis-emscripten/build_emscripten.sh
index f92b3c44..5259dc7f 100755
--- a/scripts/travis-emscripten/build_emscripten.sh
+++ b/scripts/travis-emscripten/build_emscripten.sh
@@ -50,7 +50,7 @@ sed -i 's|using gcc ;|using gcc : : /usr/local/bin/em++ ;|g' ./project-config.ja
sed -i 's|$(archiver\[1\])|/usr/local/bin/emar|g' ./tools/build/src/tools/gcc.jam
sed -i 's|$(ranlib\[1\])|/usr/local/bin/emranlib|g' ./tools/build/src/tools/gcc.jam
./b2 link=static variant=release threading=single runtime-link=static \
- thread system regex date_time chrono filesystem unit_test_framework program_options random
+ system regex filesystem unit_test_framework program_options
find . -name 'libboost*.a' -exec cp {} . \;
rm -rf b2 libs doc tools more bin.v2 status
)
@@ -61,43 +61,34 @@ echo -en 'travis_fold:start:compiling_solidity\\r'
cd $WORKSPACE
mkdir -p build
cd build
-emcmake cmake \
+cmake \
+ -DCMAKE_TOOLCHAIN_FILE=$EMSCRIPTEN/cmake/Modules/Platform/Emscripten.cmake \
-DCMAKE_BUILD_TYPE=Release \
-DEMSCRIPTEN=1 \
-DBoost_FOUND=1 \
-DBoost_USE_STATIC_LIBS=1 \
-DBoost_USE_STATIC_RUNTIME=1 \
-DBoost_INCLUDE_DIR="$WORKSPACE"/boost_1_57_0/ \
- -DBoost_CHRONO_LIBRARY="$WORKSPACE"/boost_1_57_0/libboost_chrono.a \
- -DBoost_CHRONO_LIBRARIES="$WORKSPACE"/boost_1_57_0/libboost_chrono.a \
- -DBoost_DATE_TIME_LIBRARY="$WORKSPACE"/boost_1_57_0/libboost_date_time.a \
- -DBoost_DATE_TIME_LIBRARIES="$WORKSPACE"/boost_1_57_0/libboost_date_time.a \
-DBoost_FILESYSTEM_LIBRARY="$WORKSPACE"/boost_1_57_0/libboost_filesystem.a \
-DBoost_FILESYSTEM_LIBRARIES="$WORKSPACE"/boost_1_57_0/libboost_filesystem.a \
-DBoost_PROGRAM_OPTIONS_LIBRARY="$WORKSPACE"/boost_1_57_0/libboost_program_options.a \
-DBoost_PROGRAM_OPTIONS_LIBRARIES="$WORKSPACE"/boost_1_57_0/libboost_program_options.a \
- -DBoost_RANDOM_LIBRARY="$WORKSPACE"/boost_1_57_0/libboost_random.a \
- -DBoost_RANDOM_LIBRARIES="$WORKSPACE"/boost_1_57_0/libboost_random.a \
-DBoost_REGEX_LIBRARY="$WORKSPACE"/boost_1_57_0/libboost_regex.a \
-DBoost_REGEX_LIBRARIES="$WORKSPACE"/boost_1_57_0/libboost_regex.a \
-DBoost_SYSTEM_LIBRARY="$WORKSPACE"/boost_1_57_0/libboost_system.a \
-DBoost_SYSTEM_LIBRARIES="$WORKSPACE"/boost_1_57_0/libboost_system.a \
- -DBoost_THREAD_LIBRARY="$WORKSPACE"/boost_1_57_0/libboost_thread.a \
- -DBoost_THREAD_LIBRARIES="$WORKSPACE"/boost_1_57_0/libboost_thread.a \
-DBoost_UNIT_TEST_FRAMEWORK_LIBRARY="$WORKSPACE"/boost_1_57_0/libboost_unit_test_framework.a \
-DBoost_UNIT_TEST_FRAMEWORK_LIBRARIES="$WORKSPACE"/boost_1_57_0/libboost_unit_test_framework.a \
- -DDev_DEVCORE_LIBRARY="$WORKSPACE"/solidity/build/libdevcore/libsoldevcore.a \
- -DEth_EVMASM_LIBRARY="$WORKSPACE"/solidity/build/libevmasm/libsolevmasm.a \
- -DETH_STATIC=1 -DTESTS=0 \
+ -DTESTS=0 \
..
-emmake make -j 4
+make -j 4
cd ..
-cp build/solc/soljson.js ./
mkdir -p upload
-cp soljson.js upload/
+cp build/solc/soljson.js upload/
+cp build/solc/soljson.js ./
-OUTPUT_SIZE=`ls -la build/solc/soljson.js`
+OUTPUT_SIZE=`ls -la soljson.js`
echo "Emscripten output size: $OUTPUT_SIZE"
diff --git a/test/libsolidity/SolidityNameAndTypeResolution.cpp b/test/libsolidity/SolidityNameAndTypeResolution.cpp
index 380978e8..15b06125 100644
--- a/test/libsolidity/SolidityNameAndTypeResolution.cpp
+++ b/test/libsolidity/SolidityNameAndTypeResolution.cpp
@@ -6744,6 +6744,50 @@ BOOST_AUTO_TEST_CASE(reject_interface_constructors)
CHECK_ERROR(text, TypeError, "Wrong argument count for constructor call: 1 arguments given but expected 0.");
}
+BOOST_AUTO_TEST_CASE(tight_packing_literals)
+{
+ char const* text = R"(
+ contract C {
+ function f() returns (bytes32) {
+ return keccak256(1);
+ }
+ }
+ )";
+ CHECK_WARNING(text, "The type of \"int_const 1\" was inferred as uint8.");
+ text = R"(
+ contract C {
+ function f() returns (bytes32) {
+ return keccak256(uint8(1));
+ }
+ }
+ )";
+ CHECK_SUCCESS_NO_WARNINGS(text);
+ text = R"(
+ contract C {
+ function f() returns (bytes32) {
+ return sha3(1);
+ }
+ }
+ )";
+ CHECK_WARNING(text, "The type of \"int_const 1\" was inferred as uint8.");
+ text = R"(
+ contract C {
+ function f() returns (bytes32) {
+ return sha256(1);
+ }
+ }
+ )";
+ CHECK_WARNING(text, "The type of \"int_const 1\" was inferred as uint8.");
+ text = R"(
+ contract C {
+ function f() returns (bytes32) {
+ return ripemd160(1);
+ }
+ }
+ )";
+ CHECK_WARNING(text, "The type of \"int_const 1\" was inferred as uint8.");
+}
+
BOOST_AUTO_TEST_SUITE_END()
}