diff options
author | tim@chromium.org <tim@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-10-28 22:52:59 +0000 |
---|---|---|
committer | tim@chromium.org <tim@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2008-10-28 22:52:59 +0000 |
commit | 3b8201c86ec599464a8eccb92613e77da896a6b7 (patch) | |
tree | 1aa2f15db1d2229b478d12578ea5bab384578fc3 /chrome | |
parent | 1f1c6d97f41e801e14e09fc0a6de900a2a186b56 (diff) | |
download | chromium_src-3b8201c86ec599464a8eccb92613e77da896a6b7.zip chromium_src-3b8201c86ec599464a8eccb92613e77da896a6b7.tar.gz chromium_src-3b8201c86ec599464a8eccb92613e77da896a6b7.tar.bz2 |
Fix crbug.com/3684.
Change 6053 involved making the PasswordManagerView's table model contain a vector
of stack-allocated PasswordRow that owned a heap allocated PasswordForm, but
std::vector creates temp PasswordRow's internally (on the stack) that were deleting the
PasswordForm in their dtor. This accidental vector-internal delete caused subsequent
operations on saved_signons_ to blow up.
My fix is to make PasswordRows a vector of heap allocated PasswordRow instead, and
have a STLElementDeleter member handle cleanup.
Review URL: http://codereview.chromium.org/8651
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@4103 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'chrome')
-rw-r--r-- | chrome/browser/views/password_manager_view.cc | 33 | ||||
-rw-r--r-- | chrome/browser/views/password_manager_view.h | 13 |
2 files changed, 25 insertions, 21 deletions
diff --git a/chrome/browser/views/password_manager_view.cc b/chrome/browser/views/password_manager_view.cc index 91fdc09..ab8c701 100644 --- a/chrome/browser/views/password_manager_view.cc +++ b/chrome/browser/views/password_manager_view.cc @@ -12,7 +12,6 @@ #include "chrome/views/background.h" #include "chrome/views/grid_layout.h" #include "chrome/views/native_button.h" -#include "webkit/glue/password_form.h" #include "generated_resources.h" @@ -52,17 +51,13 @@ gfx::Size MultiLabelButtons::GetPreferredSize() { return gfx::Size(pref_size_.width(), pref_size_.height()); } -//////////////////////////////////////////////////////////////////// -// PasswordManagerTableModel::PasswordRow -PasswordManagerTableModel::PasswordRow::~PasswordRow() { - delete form; -} //////////////////////////////////////////////////////////////////// // PasswordManagerTableModel PasswordManagerTableModel::PasswordManagerTableModel(Profile* profile) : observer_(NULL), pending_login_query_(NULL), + saved_signons_cleanup_(&saved_signons_), profile_(profile) { DCHECK(profile && profile->GetWebDataService(Profile::EXPLICIT_ACCESS)); } @@ -79,7 +74,7 @@ std::wstring PasswordManagerTableModel::GetText(int row, int col_id) { switch (col_id) { case IDS_PASSWORD_MANAGER_VIEW_SITE_COLUMN: // Site. - return saved_signons_[row].display_url.display_url(); + return saved_signons_[row]->display_url.display_url(); case IDS_PASSWORD_MANAGER_VIEW_USERNAME_COLUMN: // Username. return GetPasswordFormAt(row)->username_value; default: @@ -91,8 +86,8 @@ std::wstring PasswordManagerTableModel::GetText(int row, int PasswordManagerTableModel::CompareValues(int row1, int row2, int column_id) { if (column_id == IDS_PASSWORD_MANAGER_VIEW_SITE_COLUMN) { - return saved_signons_[row1].display_url.Compare( - saved_signons_[row2].display_url, GetCollator()); + return saved_signons_[row1]->display_url.Compare( + saved_signons_[row2]->display_url, GetCollator()); } return TableModel::CompareValues(row1, row2, column_id); } @@ -122,14 +117,13 @@ void PasswordManagerTableModel::OnWebDataServiceRequestDone( const WDResult<std::vector<PasswordForm*> >* r = static_cast<const WDResult<std::vector<PasswordForm*> >*>(result); std::vector<PasswordForm*> rows = r->GetValue(); - saved_signons_.clear(); - saved_signons_.resize(rows.size()); + STLDeleteElements<PasswordRows>(&saved_signons_); + saved_signons_.resize(rows.size(), NULL); std::wstring languages = profile_->GetPrefs()->GetString(prefs::kAcceptLanguages); for (size_t i = 0; i < rows.size(); ++i) { - saved_signons_[i].form = rows[i]; - saved_signons_[i].display_url = - gfx::SortedDisplayURL(rows[i]->origin, languages); + saved_signons_[i] = new PasswordRow( + gfx::SortedDisplayURL(rows[i]->origin, languages), rows[i]); } if (observer_) observer_->OnModelChanged(); @@ -144,14 +138,16 @@ void PasswordManagerTableModel::CancelLoginsQuery() { PasswordForm* PasswordManagerTableModel::GetPasswordFormAt(int row) { DCHECK(row >= 0 && row < RowCount()); - return saved_signons_[row].form; + return saved_signons_[row]->form.get(); } void PasswordManagerTableModel::ForgetAndRemoveSignon(int row) { DCHECK(row >= 0 && row < RowCount()); PasswordRows::iterator target_iter = saved_signons_.begin() + row; // Remove from DB, memory, and vector. - web_data_service()->RemoveLogin(*(target_iter->form)); + PasswordRow* password_row = *target_iter; + web_data_service()->RemoveLogin(*(password_row->form.get())); + delete password_row; saved_signons_.erase(target_iter); if (observer_) observer_->OnItemsRemoved(row, 1); @@ -160,7 +156,10 @@ void PasswordManagerTableModel::ForgetAndRemoveSignon(int row) { void PasswordManagerTableModel::ForgetAndRemoveAllSignons() { PasswordRows::iterator iter = saved_signons_.begin(); while (iter != saved_signons_.end()) { - web_data_service()->RemoveLogin(*(iter->form)); + // Remove from DB, memory, and vector. + PasswordRow* row = *iter; + web_data_service()->RemoveLogin(*(row->form.get())); + delete row; iter = saved_signons_.erase(iter); } if (observer_) diff --git a/chrome/browser/views/password_manager_view.h b/chrome/browser/views/password_manager_view.h index 1eb7ed5..8f84aa1 100644 --- a/chrome/browser/views/password_manager_view.h +++ b/chrome/browser/views/password_manager_view.h @@ -7,16 +7,18 @@ #include <vector> +#include "base/scoped_ptr.h" #include "chrome/browser/webdata/web_data_service.h" +#include "chrome/common/stl_util-inl.h" #include "chrome/common/gfx/url_elider.h" #include "chrome/views/dialog_delegate.h" #include "chrome/views/label.h" #include "chrome/views/native_button.h" #include "chrome/views/table_view.h" #include "chrome/views/window.h" +#include "webkit/glue/password_form.h" class Profile; -struct PasswordForm; class PasswordManagerTableModel : public views::TableModel, public WebDataServiceConsumer { @@ -51,13 +53,15 @@ class PasswordManagerTableModel : public views::TableModel, // Wraps the PasswordForm from the database and caches the display URL for // quick sorting. struct PasswordRow { - ~PasswordRow(); + PasswordRow(const gfx::SortedDisplayURL& url, PasswordForm* password_form) + : display_url(url), form(password_form) { + } // Contains the URL that is displayed along with the gfx::SortedDisplayURL display_url; // The underlying PasswordForm. We own this. - PasswordForm* form; + scoped_ptr<PasswordForm> form; }; // Cancel any pending login query involving a callback. @@ -75,8 +79,9 @@ class PasswordManagerTableModel : public views::TableModel, WebDataService::Handle pending_login_query_; // The set of passwords we're showing. - typedef std::vector<PasswordRow> PasswordRows; + typedef std::vector<PasswordRow*> PasswordRows; PasswordRows saved_signons_; + STLElementDeleter<PasswordRows> saved_signons_cleanup_; Profile* profile_; |