SQL/ODBC: fix some users of toSQLTCHAR() to not assume identical UTF-8/16/32 string lengths

We already fixed the implementation of toSQLTCHAR() in
66767eea46 to not assume that a UTF-8 or
UTF-32-encoded string has the same number of code points as the
equivalent UTF-16 string, but it turns out that users of the function,
as well as other code, also failed to account for this.

This patch fixes callers of toSQLTCHAR() to use

    const auto encoded = toSQLTCHAR(s);
    ~~~ use encoded.data(), encoded.size() ~~~

(except we can't make `encoded` const, because the SQL API isn't
const-correct and takes void* instead of const void*) instead of the
anti-pattern

   ~~~ use toSQLTCHAR(s).data(), s.size() ~~~

As a drive-by:
- Extract Method qt_string_SQLSetConnectAttr()
  - skipping an unneeded .utf16() call (a NUL-terminated string is not
    required for calling toSQLTCHAR())
- de-duplicate some code in exec()
  - and make a comment there slightly more informative
- replace
  - NULL with nullptr
  - size() == 0 with isEmpty()
  - C-style with constructor-style casts

Pick-to: 6.5 6.4 6.2 5.15
Change-Id: I3696381d0a93af8861ce2b7915f212d9e5e9a243
Reviewed-by: Thiago Macieira <thiago.macieira@intel.com>
This commit is contained in:
Marc Mutz 2023-01-30 15:37:13 +01:00
parent e73389874d
commit 46af1fe49f

View File

@ -745,6 +745,14 @@ QChar QODBCDriverPrivate::quoteChar()
return quote; return quote;
} }
static SQLRETURN qt_string_SQLSetConnectAttr(SQLHDBC handle, SQLINTEGER attr, const QString &val)
{
auto encoded = toSQLTCHAR(val);
return SQLSetConnectAttr(handle, attr,
encoded.data(),
SQLINTEGER(encoded.size() * sizeof(SQLTCHAR))); // size in bytes
}
bool QODBCDriverPrivate::setConnectionOptions(const QString& connOpts) bool QODBCDriverPrivate::setConnectionOptions(const QString& connOpts)
{ {
@ -780,10 +788,7 @@ bool QODBCDriverPrivate::setConnectionOptions(const QString& connOpts)
v = val.toUInt(); v = val.toUInt();
r = SQLSetConnectAttr(hDbc, SQL_ATTR_LOGIN_TIMEOUT, (SQLPOINTER) size_t(v), 0); r = SQLSetConnectAttr(hDbc, SQL_ATTR_LOGIN_TIMEOUT, (SQLPOINTER) size_t(v), 0);
} else if (opt.toUpper() == "SQL_ATTR_CURRENT_CATALOG"_L1) { } else if (opt.toUpper() == "SQL_ATTR_CURRENT_CATALOG"_L1) {
val.utf16(); // 0 terminate r = qt_string_SQLSetConnectAttr(hDbc, SQL_ATTR_CURRENT_CATALOG, val);
r = SQLSetConnectAttr(hDbc, SQL_ATTR_CURRENT_CATALOG,
toSQLTCHAR(val).data(),
SQLINTEGER(val.length() * sizeof(SQLTCHAR)));
} else if (opt.toUpper() == "SQL_ATTR_METADATA_ID"_L1) { } else if (opt.toUpper() == "SQL_ATTR_METADATA_ID"_L1) {
if (val.toUpper() == "SQL_TRUE"_L1) { if (val.toUpper() == "SQL_TRUE"_L1) {
v = SQL_TRUE; v = SQL_TRUE;
@ -798,10 +803,7 @@ bool QODBCDriverPrivate::setConnectionOptions(const QString& connOpts)
v = val.toUInt(); v = val.toUInt();
r = SQLSetConnectAttr(hDbc, SQL_ATTR_PACKET_SIZE, (SQLPOINTER) size_t(v), 0); r = SQLSetConnectAttr(hDbc, SQL_ATTR_PACKET_SIZE, (SQLPOINTER) size_t(v), 0);
} else if (opt.toUpper() == "SQL_ATTR_TRACEFILE"_L1) { } else if (opt.toUpper() == "SQL_ATTR_TRACEFILE"_L1) {
val.utf16(); // 0 terminate r = qt_string_SQLSetConnectAttr(hDbc, SQL_ATTR_TRACEFILE, val);
r = SQLSetConnectAttr(hDbc, SQL_ATTR_TRACEFILE,
toSQLTCHAR(val).data(),
SQLINTEGER(val.length() * sizeof(SQLTCHAR)));
} else if (opt.toUpper() == "SQL_ATTR_TRACE"_L1) { } else if (opt.toUpper() == "SQL_ATTR_TRACE"_L1) {
if (val.toUpper() == "SQL_OPT_TRACE_OFF"_L1) { if (val.toUpper() == "SQL_OPT_TRACE_OFF"_L1) {
v = SQL_OPT_TRACE_OFF; v = SQL_OPT_TRACE_OFF;
@ -1004,9 +1006,12 @@ bool QODBCResult::reset (const QString& query)
return false; return false;
} }
r = SQLExecDirect(d->hStmt, {
toSQLTCHAR(query).data(), auto encoded = toSQLTCHAR(query);
(SQLINTEGER) query.length()); r = SQLExecDirect(d->hStmt,
encoded.data(),
SQLINTEGER(encoded.size()));
}
if (r != SQL_SUCCESS && r != SQL_SUCCESS_WITH_INFO && r!= SQL_NO_DATA) { if (r != SQL_SUCCESS && r != SQL_SUCCESS_WITH_INFO && r!= SQL_NO_DATA) {
setLastError(qMakeError(QCoreApplication::translate("QODBCResult", setLastError(qMakeError(QCoreApplication::translate("QODBCResult",
"Unable to execute statement"), QSqlError::StatementError, d)); "Unable to execute statement"), QSqlError::StatementError, d));
@ -1355,9 +1360,12 @@ bool QODBCResult::prepare(const QString& query)
return false; return false;
} }
r = SQLPrepare(d->hStmt, {
toSQLTCHAR(query).data(), auto encoded = toSQLTCHAR(query);
(SQLINTEGER) query.length()); r = SQLPrepare(d->hStmt,
encoded.data(),
SQLINTEGER(encoded.size()));
}
if (r != SQL_SUCCESS) { if (r != SQL_SUCCESS) {
setLastError(qMakeError(QCoreApplication::translate("QODBCResult", setLastError(qMakeError(QCoreApplication::translate("QODBCResult",
@ -1385,7 +1393,7 @@ bool QODBCResult::exec()
SQLCloseCursor(d->hStmt); SQLCloseCursor(d->hStmt);
QVariantList &values = boundValues(); QVariantList &values = boundValues();
QByteArrayList tmpStorage(values.count(), QByteArray()); // holds temporary buffers QByteArrayList tmpStorage(values.count(), QByteArray()); // targets for SQLBindParameter()
QVarLengthArray<SQLLEN, 32> indicators(values.count()); QVarLengthArray<SQLLEN, 32> indicators(values.count());
memset(indicators.data(), 0, indicators.size() * sizeof(SQLLEN)); memset(indicators.data(), 0, indicators.size() * sizeof(SQLLEN));
@ -1600,36 +1608,36 @@ bool QODBCResult::exec()
case QMetaType::QString: case QMetaType::QString:
if (d->unicode) { if (d->unicode) {
QByteArray &ba = tmpStorage[i]; QByteArray &ba = tmpStorage[i];
QString str = val.toString(); {
const auto encoded = toSQLTCHAR(val.toString());
ba = QByteArray(reinterpret_cast<const char *>(encoded.data()),
encoded.size() * sizeof(SQLTCHAR));
}
if (*ind != SQL_NULL_DATA) if (*ind != SQL_NULL_DATA)
*ind = str.length() * sizeof(SQLTCHAR); *ind = ba.size();
const qsizetype strSize = str.length() * sizeof(SQLTCHAR);
if (bindValueType(i) & QSql::Out) { if (bindValueType(i) & QSql::Out) {
const QVarLengthArray<SQLTCHAR> a(toSQLTCHAR(str));
ba = QByteArray((const char *)a.constData(), int(a.size() * sizeof(SQLTCHAR)));
r = SQLBindParameter(d->hStmt, r = SQLBindParameter(d->hStmt,
i + 1, i + 1,
qParamType[bindValueType(i) & QSql::InOut], qParamType[bindValueType(i) & QSql::InOut],
SQL_C_TCHAR, SQL_C_TCHAR,
strSize > 254 ? SQL_WLONGVARCHAR : SQL_WVARCHAR, ba.size() > 254 ? SQL_WLONGVARCHAR : SQL_WVARCHAR,
0, // god knows... don't change this! 0, // god knows... don't change this!
0, 0,
ba.data(), const_cast<char *>(ba.constData()), // don't detach
ba.size(), ba.size(),
ind); ind);
break; break;
} }
ba = QByteArray(reinterpret_cast<const char *>(toSQLTCHAR(str).constData()),
int(strSize));
r = SQLBindParameter(d->hStmt, r = SQLBindParameter(d->hStmt,
i + 1, i + 1,
qParamType[bindValueType(i) & QSql::InOut], qParamType[bindValueType(i) & QSql::InOut],
SQL_C_TCHAR, SQL_C_TCHAR,
strSize > 254 ? SQL_WLONGVARCHAR : SQL_WVARCHAR, ba.size() > 254 ? SQL_WLONGVARCHAR : SQL_WVARCHAR,
strSize, ba.size(),
0, 0,
const_cast<char *>(ba.constData()), const_cast<char *>(ba.constData()), // don't detach
ba.size(), ba.size(),
ind); ind);
break; break;
@ -1991,14 +1999,16 @@ bool QODBCDriver::open(const QString & db,
SQLSMALLINT cb; SQLSMALLINT cb;
QVarLengthArray<SQLTCHAR> connOut(1024); QVarLengthArray<SQLTCHAR> connOut(1024);
memset(connOut.data(), 0, connOut.size() * sizeof(SQLTCHAR)); memset(connOut.data(), 0, connOut.size() * sizeof(SQLTCHAR));
r = SQLDriverConnect(d->hDbc, {
NULL, auto encoded = toSQLTCHAR(connQStr);
toSQLTCHAR(connQStr).data(), r = SQLDriverConnect(d->hDbc,
(SQLSMALLINT)connQStr.length(), nullptr,
connOut.data(), encoded.data(), SQLSMALLINT(encoded.size()),
1024, connOut.data(),
&cb, 1024,
/*SQL_DRIVER_NOPROMPT*/0); &cb,
/*SQL_DRIVER_NOPROMPT*/0);
}
if (r != SQL_SUCCESS && r != SQL_SUCCESS_WITH_INFO) { if (r != SQL_SUCCESS && r != SQL_SUCCESS_WITH_INFO) {
setLastError(qMakeError(tr("Unable to connect"), QSqlError::ConnectionError, d)); setLastError(qMakeError(tr("Unable to connect"), QSqlError::ConnectionError, d));
@ -2377,17 +2387,15 @@ QStringList QODBCDriver::tables(QSql::TableType type) const
if (tableType.isEmpty()) if (tableType.isEmpty())
return tl; return tl;
QString joinedTableTypeString = tableType.join(u','); {
auto joinedTableTypeString = toSQLTCHAR(tableType.join(u','));
r = SQLTables(hStmt, r = SQLTables(hStmt,
NULL, nullptr, 0,
0, nullptr, 0,
NULL, nullptr, 0,
0, joinedTableTypeString.data(), joinedTableTypeString.size());
NULL, }
0,
toSQLTCHAR(joinedTableTypeString).data(),
joinedTableTypeString.length() /* characters, not bytes */);
if (r != SQL_SUCCESS) if (r != SQL_SUCCESS)
qSqlWarning("QODBCDriver::tables Unable to execute table list"_L1, d); qSqlWarning("QODBCDriver::tables Unable to execute table list"_L1, d);
@ -2460,28 +2468,30 @@ QSqlIndex QODBCDriver::primaryIndex(const QString& tablename) const
SQL_ATTR_CURSOR_TYPE, SQL_ATTR_CURSOR_TYPE,
(SQLPOINTER)SQL_CURSOR_FORWARD_ONLY, (SQLPOINTER)SQL_CURSOR_FORWARD_ONLY,
SQL_IS_UINTEGER); SQL_IS_UINTEGER);
r = SQLPrimaryKeys(hStmt, {
catalog.length() == 0 ? NULL : toSQLTCHAR(catalog).data(), auto c = toSQLTCHAR(catalog);
catalog.length(), auto s = toSQLTCHAR(schema);
schema.length() == 0 ? NULL : toSQLTCHAR(schema).data(), auto t = toSQLTCHAR(table);
schema.length(), r = SQLPrimaryKeys(hStmt,
toSQLTCHAR(table).data(), catalog.isEmpty() ? nullptr : c.data(), c.size(),
table.length() /* in characters, not in bytes */); schema.isEmpty() ? nullptr : s.data(), s.size(),
t.data(), t.size());
}
// if the SQLPrimaryKeys() call does not succeed (e.g the driver // if the SQLPrimaryKeys() call does not succeed (e.g the driver
// does not support it) - try an alternative method to get hold of // does not support it) - try an alternative method to get hold of
// the primary index (e.g MS Access and FoxPro) // the primary index (e.g MS Access and FoxPro)
if (r != SQL_SUCCESS) { if (r != SQL_SUCCESS) {
r = SQLSpecialColumns(hStmt, auto c = toSQLTCHAR(catalog);
SQL_BEST_ROWID, auto s = toSQLTCHAR(schema);
catalog.length() == 0 ? NULL : toSQLTCHAR(catalog).data(), auto t = toSQLTCHAR(table);
catalog.length(), r = SQLSpecialColumns(hStmt,
schema.length() == 0 ? NULL : toSQLTCHAR(schema).data(), SQL_BEST_ROWID,
schema.length(), catalog.isEmpty() ? nullptr : c.data(), c.size(),
toSQLTCHAR(table).data(), schema.isEmpty() ? nullptr : s.data(), s.size(),
table.length(), t.data(), t.size(),
SQL_SCOPE_CURROW, SQL_SCOPE_CURROW,
SQL_NULLABLE); SQL_NULLABLE);
if (r != SQL_SUCCESS) { if (r != SQL_SUCCESS) {
qSqlWarning("QODBCDriver::primaryIndex: Unable to execute primary key list"_L1, d); qSqlWarning("QODBCDriver::primaryIndex: Unable to execute primary key list"_L1, d);
@ -2562,15 +2572,17 @@ QSqlRecord QODBCDriver::record(const QString& tablename) const
SQL_ATTR_CURSOR_TYPE, SQL_ATTR_CURSOR_TYPE,
(SQLPOINTER)SQL_CURSOR_FORWARD_ONLY, (SQLPOINTER)SQL_CURSOR_FORWARD_ONLY,
SQL_IS_UINTEGER); SQL_IS_UINTEGER);
r = SQLColumns(hStmt, {
catalog.length() == 0 ? NULL : toSQLTCHAR(catalog).data(), auto c = toSQLTCHAR(catalog);
catalog.length(), auto s = toSQLTCHAR(schema);
schema.length() == 0 ? NULL : toSQLTCHAR(schema).data(), auto t = toSQLTCHAR(table);
schema.length(), r = SQLColumns(hStmt,
toSQLTCHAR(table).data(), catalog.isEmpty() ? nullptr : c.data(), c.size(),
table.length(), schema.isEmpty() ? nullptr : s.data(), s.size(),
NULL, t.data(), t.size(),
0); nullptr,
0);
}
if (r != SQL_SUCCESS) if (r != SQL_SUCCESS)
qSqlWarning("QODBCDriver::record: Unable to execute column list"_L1, d); qSqlWarning("QODBCDriver::record: Unable to execute column list"_L1, d);